annotate dyncallback/dyncall_thunk_ppc64.c @ 0:3e629dc19168

initial from svn dyncall-1745
author Daniel Adler
date Thu, 19 Mar 2015 22:24:28 +0100
parents
children 7ca57dbefed4
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
1 /*
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
2
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
3 Package: dyncall
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
4 Library: dyncallback
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
5 File: dyncallback/dyncall_thunk_ppc64.c
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
6 Description: Thunk - Implementation for ppc64
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
7 License:
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
8
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
9 Copyright (c) 2014-2015 Masanori Mitsugi <mitsugi@linux.vnet.ibm.com>
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
10
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
11 Permission to use, copy, modify, and distribute this software for any
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
12 purpose with or without fee is hereby granted, provided that the above
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
13 copyright notice and this permission notice appear in all copies.
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
14
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
15 THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
16 WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
17 MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
18 ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
19 WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
20 ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
21 OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
22
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
23 */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
24
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
25 #include "dyncall_thunk.h"
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
26
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
27 #define HIST16(x) (unsigned short)((((unsigned long)(x))>>48UL) & 0xffff)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
28 #define HIER16(x) (unsigned short)((((unsigned long)(x))>>32UL) & 0xffff)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
29 #define HI16(x) (unsigned short)((((unsigned long)(x))>>16UL) & 0xffff)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
30 #define LO16(x) (unsigned short)( ((unsigned long)(x)) & 0xffff)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
31
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
32 void dcbInitThunk(DCThunk* p, void (*entry)())
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
33 {
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
34 #if DC__ABI_PPC64_ELF_V == 2
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
35 /*
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
36 ppc64 thunk code:
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
37 lis r11, HIST16(p)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
38 ori r11,r11, HIER16(p)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
39 rldicr r11,r11,32,31
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
40 oris r11,r11, HI16(p)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
41 ori r11,r11, LO16(p)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
42 ld r12,40(r11)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
43 mtctr r12
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
44 bctr
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
45 */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
46
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
47 p->code_load_hist = 0x3d60U; /* lis r11,HIST16(p) */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
48 p->addr_self_hist = HIST16(p);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
49 p->code_load_hier = 0x616bU; /* ori r11,r11, HIER16(p) */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
50 p->addr_self_hier = HIER16(p);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
51 p->code_rot = 0x796b07c6U; /* rldicr r11,r11,32,31 */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
52 p->code_load_hi = 0x656bU; /* oris r11,r11, HI16(p) */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
53 p->addr_self_hi = HI16(p);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
54 p->code_load_lo = 0x616bU; /* ori r11,r11, LO16(p) */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
55 p->addr_self_lo = LO16(p);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
56 p->code_jump[0] = 0xe98b0028U; /* ld r12,40(r11) */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
57 p->code_jump[1] = 0x7d8903a6U; /* mtclr r12 */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
58 p->code_jump[2] = 0x4e800420U; /* bctr */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
59 p->addr_entry = (void *)(entry);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
60
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
61 #else
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
62 /*
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
63 ppc64 thunk code:
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
64 oris r11, r2, HI16(p)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
65 ori r11,r11, LO16(p)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
66 ld r12,48(r11)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
67 ld r2,56(r11)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
68 mtctr r12
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
69 bctr
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
70 */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
71
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
72 p->thunk_entry = (void *)&(p->code_load_hi);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
73 p->toc_thunk = ((long)(p->thunk_entry) & 0xffffffff00000000UL);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
74
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
75 p->code_load_hi = 0x644bU; /* oris r11, r2, HI16(p) */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
76 p->addr_self_hi = HI16(p);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
77 p->code_load_lo = 0x616bU; /* ori r11,r11, LO16(p) */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
78 p->addr_self_lo = LO16(p);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
79 p->code_jump[0] = 0xe98b0030U; /* ld r12,48(r11) */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
80 p->code_jump[1] = 0xe84b0038U; /* ld r2,56(r11) */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
81 p->code_jump[2] = 0x7d8903a6U; /* mtclr r12 */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
82 p->code_jump[3] = 0x4e800420U; /* bctr */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
83 p->addr_entry = (void *)*((long *)entry);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
84 p->toc_entry = *((long *)(entry + 8));
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
85 #endif
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
86 }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
87