annotate dyncallback/dyncall_args_arm32_arm.c @ 38:fb416abb2059

- armhf callback support (not all working, yet)
author cslag
date Fri, 18 Dec 2015 19:07:17 +0100
parents 3e629dc19168
children f01895437921
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_args_arm32_arm.c
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
6 Description: Callback's Arguments VM - Implementation for ARM32 (ARM mode)
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) 2007-2015 Daniel Adler <dadler@uni-goettingen.de>,
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
10 Tassilo Philipp <tphilipp@potion-studios.com>
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
11
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
12 Permission to use, copy, modify, and distribute this software for any
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
13 purpose with or without fee is hereby granted, provided that the above
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
14 copyright notice and this permission notice appear in all copies.
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
15
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
16 THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
17 WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
18 MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
19 ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
20 WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
21 ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
22 OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
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
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
26
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
27
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
28 #include "dyncall_args_arm32_arm.h"
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
29
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
30
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
31 static void arm_align_64(DCArgs* args)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
32 {
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
33 /* Look at signature to see if current calling convention needs alignment */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
34 /* or not (e.g. EABI has different alignment). If nothing specified, fall */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
35 /* back to default behaviour for this platform. */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
36 /* @@@ check signature string */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
37
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
38 int sig =
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
39 #if defined(DC__ABI_ARM_EABI)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
40 0; /* EABI */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
41 #else
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
42 1; /* ATPCS */
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
43 #endif
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
44 if(sig == 0) {
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
45 if(args->reg_count < 4)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
46 args->reg_count = (args->reg_count+1)&~1;
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
47 if(args->reg_count >= 4 && (int)args->stack_ptr & 4)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
48 ++args->stack_ptr;
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
49 }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
50 }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
51
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
52
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
53 static void* arm_word(DCArgs* args)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
54 {
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
55 if(args->reg_count < 4)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
56 return &args->reg_data[args->reg_count++];
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
57 else
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
58 return (void*)args->stack_ptr++;
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
59 }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
60
38
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
61 static DCfloat arm_float(DCArgs* args)
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
62 {
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
63 #if defined(DC__ABI_ARM_HF)
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
64 DCfloat f;
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
65 if(args->freg_count < 16) {
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
66 f = args->f[args->freg_count++];
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
67
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
68 /* if freg_count was odd, sync with dreg_count */
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
69 if(!(args->freg_count & 1) && (args->freg_count < args->dreg_count))
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
70 args->freg_count = args->dreg_count;
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
71
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
72 return f;
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
73 }
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
74 #endif
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
75 return *(DCfloat*)arm_word(args);
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
76 }
0
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
77
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
78 static DCdouble arm_double(DCArgs* args)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
79 {
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
80 union {
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
81 DCdouble d;
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
82 DClong l[2];
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
83 } d;
38
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
84 #if defined(DC__ABI_ARM_HF)
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
85 if(args->dreg_count < args->freg_count)
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
86 args->dreg_count = (args->freg_count+1)&0x1e; /* clear last bit, counter won't be higher than 16, anyways */
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
87
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
88 if(args->dreg_count < 16) {
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
89 d.d = *(DCdouble*)&args->f[args->dreg_count];
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
90 args->dreg_count += 2;
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
91
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
92 /* freg_count is either odd (pointing to a gap), or always the same as dreg_count */
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
93 if(!(args->freg_count & 1))
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
94 args->freg_count = args->dreg_count;
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
95 return d.d;
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
96 }
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
97 #endif
0
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
98 arm_align_64(args);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
99 d.l[0] = *(DClong*)arm_word(args);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
100 d.l[1] = *(DClong*)arm_word(args);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
101 return d.d;
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
102 }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
103
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
104 static DClonglong arm_longlong(DCArgs* args)
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
105 {
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
106 union {
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
107 DClonglong ll;
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
108 DClong l[2];
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
109 } ll;
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
110 arm_align_64(args);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
111 ll.l[0] = *(DClong*)arm_word(args);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
112 ll.l[1] = *(DClong*)arm_word(args);
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
113 return ll.ll;
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
114 }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
115
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
116
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
117
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
118 // ----------------------------------------------------------------------------
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
119 // C API implementation:
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
120
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
121
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
122 // base operations:
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
123
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
124 DClonglong dcbArgLongLong (DCArgs* p) { return arm_longlong(p); }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
125 DClong dcbArgLong (DCArgs* p) { return *(DClong*)arm_word(p); }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
126 DCint dcbArgInt (DCArgs* p) { return (DCint) dcbArgLong(p); }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
127 DCchar dcbArgChar (DCArgs* p) { return (DCchar) dcbArgLong(p); }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
128 DCshort dcbArgShort (DCArgs* p) { return (DCshort) dcbArgLong(p); }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
129 DCbool dcbArgBool (DCArgs* p) { return (dcbArgLong(p) == 0) ? 0 : 1; }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
130
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
131 DCuint dcbArgUInt (DCArgs* p) { return (DCuint) dcbArgInt(p); }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
132 DCuchar dcbArgUChar (DCArgs* p) { return (DCuchar) dcbArgChar(p); }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
133 DCushort dcbArgUShort (DCArgs* p) { return (DCushort) dcbArgShort(p); }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
134 DCulong dcbArgULong (DCArgs* p) { return (DCulong) dcbArgLong(p); }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
135 DCulonglong dcbArgULongLong(DCArgs* p) { return (DCulonglong)dcbArgLongLong(p); }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
136
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
137
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
138 DCpointer dcbArgPointer (DCArgs* p) { return (DCpointer) dcbArgLong(p); }
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
139
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
140 DCdouble dcbArgDouble (DCArgs* p) { return arm_double(p); }
38
fb416abb2059 - armhf callback support (not all working, yet)
cslag
parents: 0
diff changeset
141 DCfloat dcbArgFloat (DCArgs* p) { return arm_float(p); }
0
3e629dc19168 initial from svn dyncall-1745
Daniel Adler
parents:
diff changeset
142