Release 0.4.0
[wine] / if1632 / relay.c
1 static char RCSId[] = "$Id: relay.c,v 1.2 1993/07/04 04:04:21 root Exp root $";
2 static char Copyright[] = "Copyright  Robert J. Amstadt, 1993";
3
4 #include <stdio.h>
5 #include <stdlib.h>
6 #include <sys/types.h>
7 #include <sys/stat.h>
8 #include <fcntl.h>
9 #include <unistd.h>
10 #ifdef linux
11 #include <linux/unistd.h>
12 #include <linux/head.h>
13 #include <linux/ldt.h>
14 #include <linux/segment.h>
15 #endif
16 #include <errno.h>
17 #include "neexe.h"
18 #include "segmem.h"
19 #include "prototypes.h"
20 #include "dlls.h"
21
22 #define N_BUILTINS      8
23
24 struct dll_name_table_entry_s dll_builtin_table[N_BUILTINS] =
25 {
26     { "KERNEL",  KERNEL_table,  410, 1 },
27     { "USER",    USER_table,    540, 2 },
28     { "GDI",     GDI_table,     490, 3 },
29     { "UNIXLIB", UNIXLIB_table,  10, 4 },
30     { "WIN87EM", WIN87EM_table,  10, 5 },
31     { "SHELL",   SHELL_table,   256, 6 },
32     { "SOUND",   SOUND_table,    20, 7 },
33     { "KEYBOARD",KEYBOARD_table,137, 8 },
34 };
35
36 unsigned short *Stack16Frame;
37
38 extern unsigned long  IF1632_Saved16_esp;
39 extern unsigned long  IF1632_Saved16_ebp;
40 extern unsigned short IF1632_Saved16_ss;
41
42 /**********************************************************************
43  *                                      DLLRelay
44  *
45  * We get a stack frame pointer to data that looks like this:
46  *
47  *   Hex Offset Contents
48  *   ---------- -------
49  *      +00     previous saved_16ss
50  *      +02     previous saved_16ebp
51  *      +06     previous saved_16esp
52  *      +0A     16-bit es
53  *      +0C     16-bit ds
54  *      +0E     16-bit ebp
55  *      +12     length of 16-bit arguments
56  *      +14     16-bit ip
57  *      +16     16-bit cs
58  *      +18     arguments
59  */
60 int
61 DLLRelay(unsigned int func_num, unsigned int seg_off)
62 {
63     struct dll_table_entry_s *dll_p;
64     unsigned int segment;
65     unsigned int offset;
66     unsigned int dll_id;
67     unsigned int ordinal;
68     int arg_table[DLL_MAX_ARGS];
69     void *arg_ptr;
70     int (*func_ptr)();
71     int i;
72     int ret_val;
73     
74     /*
75      * Determine address of arguments.
76      */
77     Stack16Frame = (unsigned short *) seg_off;
78     arg_ptr = (void *) (seg_off + 0x18);
79
80     /*
81      * Extract the DLL number and ordinal number.
82      */
83     dll_id  = ((func_num >> 16) & 0xffff) - 1;
84     ordinal = func_num & 0xffff;
85     dll_p   = &dll_builtin_table[dll_id].dll_table[ordinal];
86
87 #ifdef DEBUG_RELAY
88     {
89         unsigned int *ret_addr;
90         unsigned short *stack_p;
91         
92         ret_addr = (unsigned int *) ((char *) seg_off + 0x14);
93         printf("Calling %s (%s.%d), 16-bit stack at %04x:%04x, ",
94                dll_p->export_name,
95                dll_builtin_table[dll_id].dll_name, ordinal,
96                seg_off >> 16, seg_off & 0xffff);
97         printf("return to %08x\n", *ret_addr);
98         printf("  ESP %08x, EBP %08x, SS %04x\n", 
99                IF1632_Saved16_esp, IF1632_Saved16_ebp,
100                IF1632_Saved16_ss);
101
102         if (strcmp("GetMessage", dll_p->export_name) == 0 &&
103             seg_off == 0x00972526 &&
104             *ret_addr == 0x004700cd &&
105             IF1632_Saved16_esp == 0x2526 &&
106             IF1632_Saved16_ebp == 0x2534 &&
107             IF1632_Saved16_ss  == 0x0097)
108             printf("ACK!!\n");
109
110 #if 0
111         IF1632_Saved16_esp &= 0x0000ffff;
112         IF1632_Saved16_ebp &= 0x0000ffff;
113 #endif
114
115 #ifdef DEBUG_STACK
116         stack_p = (unsigned short *) seg_off;
117         for (i = 0; i < 24; i++, stack_p++)
118         {
119             printf("%04x ", *stack_p);
120             if ((i & 7) == 7)
121                 printf("\n");
122         }
123         printf("\n");
124 #endif /* DEBUG_STACK */
125     }
126 #endif /* DEBUG_RELAY */
127
128     /*
129      * Make sure we have a handler defined for this call.
130      */
131     if (dll_p->handler == NULL)
132     {
133         char buffer[100];
134         
135         sprintf(buffer, "No handler for routine %s.%d", 
136                 dll_builtin_table[dll_id].dll_name, ordinal);
137         myerror(buffer);
138     }
139     func_ptr = dll_p->handler;
140
141     /*
142      * OK, special case.  If the handler is define as taking no arguments
143      * then pass the address of the arguments on the 16-bit stack to the
144      * handler.  It will just ignore the pointer if it really takes no
145      * arguments.  This allows us to write slightly faster library routines
146      * if we choose.
147      */
148     if (dll_p->n_args == 0)
149         return (*func_ptr)(arg_ptr);
150
151     /*
152      * Getting this far means we need to convert the 16-bit argument stack.
153      */
154     for (i = 0; i < dll_p->n_args; i++)
155     {
156         short *sp;
157         int *ip;
158         
159         offset = dll_p->args[i].dst_arg;
160
161         switch (dll_p->args[i].src_type)
162         {
163           case DLL_ARGTYPE_SIGNEDWORD:
164             sp = (short *) ((char *) arg_ptr + offset);
165             arg_table[i] = *sp;
166             break;
167             
168           case DLL_ARGTYPE_WORD:
169             sp = (short *) ((char *) arg_ptr + offset);
170             arg_table[i] = (int) *sp & 0xffff;
171             break;
172             
173           case DLL_ARGTYPE_LONG:
174           case DLL_ARGTYPE_FARPTR:
175             ip = (int *) ((char *) arg_ptr + offset);
176             arg_table[i] = *ip;
177             break;
178         }
179     }
180
181     /*
182      * Call the handler
183      */
184     ret_val = (*func_ptr)(arg_table[0], arg_table[1], arg_table[2], 
185                           arg_table[3], arg_table[4], arg_table[5], 
186                           arg_table[6], arg_table[7], arg_table[8], 
187                           arg_table[9], arg_table[10], arg_table[11],
188                           arg_table[12], arg_table[13], arg_table[14], 
189                           arg_table[15]);
190
191 #ifdef DEBUG_RELAY
192     printf("Returning %08.8x from %s (%s.%d)\n",
193            ret_val,
194            dll_p->export_name,
195            dll_builtin_table[dll_id].dll_name, ordinal);
196 #endif
197
198     return ret_val;
199 }
200
201 /**********************************************************************
202  *                                      FindDLLTable
203  */
204 struct  dll_table_entry_s *
205 FindDLLTable(char *dll_name)
206 {
207     int i;
208
209     for (i = 0; i < N_BUILTINS; i++)
210         if (strcmp(dll_builtin_table[i].dll_name, dll_name) == 0)
211             return dll_builtin_table[i].dll_table;
212     
213     return NULL;
214 }
215
216 /**********************************************************************
217  *                                      FindOrdinalFromName
218  */
219 int
220 FindOrdinalFromName(struct dll_table_entry_s *dll_table, char *func_name)
221 {
222     int i, limit;
223
224     for (i = 0; i < N_BUILTINS; i++)
225         if (dll_table == dll_builtin_table[i].dll_table)
226             break;
227     
228     if (i == N_BUILTINS)
229         return 0;
230
231     limit = dll_builtin_table[i].dll_table_length;
232     for (i = 0; i < limit; i++)
233         if (strcasecmp(dll_table[i].export_name, func_name) == 0)
234             return i;
235     
236     return 0;
237 }
238 /**********************************************************************
239  *                                      ReturnArg
240  */
241 int
242 ReturnArg(int arg)
243 {
244     return arg;
245 }
246
247 #ifdef WINESTAT
248 void winestat(){
249         int i, j;
250         double perc;
251         int used, implemented;
252         int tused, timplemented;
253         struct dll_table_entry_s *table;
254
255         tused = 0;
256         timplemented = 0;
257     for (i = 0; i < N_BUILTINS; i++) {
258             table = dll_builtin_table[i].dll_table;
259             used = 0;
260             implemented = 0;
261             for(j=0; j < dll_builtin_table[i].dll_table_length; j++) {
262                     if(table[j].used){
263                             used++;
264                             if (table[j].handler) implemented++;
265                             else 
266                                     printf("%s.%d\n",
267                                            dll_builtin_table[i].dll_name,
268                                            j);
269                     };
270             };
271             tused += used;
272             timplemented += implemented;
273             if(used)
274                     perc = implemented * 100.00 / used;
275             else
276                     perc = 0.0;
277             printf("%s: %d %d %3.1f\n", dll_builtin_table[i].dll_name, implemented, used, perc);
278     };
279         perc = timplemented * 100.00 / tused;
280         printf("TOTAL: %d %d %3.1f\n",timplemented, tused, perc);
281 }
282 #endif /* WINESTAT */