]> git.proxmox.com Git - qemu.git/blob - target-i386/ops_template.h
helper_lret_protected fix for kqemu (Paul Brook)
[qemu.git] / target-i386 / ops_template.h
1 /*
2 * i386 micro operations (included several times to generate
3 * different operand sizes)
4 *
5 * Copyright (c) 2003 Fabrice Bellard
6 *
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2 of the License, or (at your option) any later version.
11 *
12 * This library 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 GNU
15 * Lesser General Public License for more details.
16 *
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 */
21 #define DATA_BITS (1 << (3 + SHIFT))
22 #define SHIFT_MASK (DATA_BITS - 1)
23 #define SIGN_MASK (((target_ulong)1) << (DATA_BITS - 1))
24 #if DATA_BITS <= 32
25 #define SHIFT1_MASK 0x1f
26 #else
27 #define SHIFT1_MASK 0x3f
28 #endif
29
30 #if DATA_BITS == 8
31 #define SUFFIX b
32 #define DATA_TYPE uint8_t
33 #define DATA_STYPE int8_t
34 #define DATA_MASK 0xff
35 #elif DATA_BITS == 16
36 #define SUFFIX w
37 #define DATA_TYPE uint16_t
38 #define DATA_STYPE int16_t
39 #define DATA_MASK 0xffff
40 #elif DATA_BITS == 32
41 #define SUFFIX l
42 #define DATA_TYPE uint32_t
43 #define DATA_STYPE int32_t
44 #define DATA_MASK 0xffffffff
45 #elif DATA_BITS == 64
46 #define SUFFIX q
47 #define DATA_TYPE uint64_t
48 #define DATA_STYPE int64_t
49 #define DATA_MASK 0xffffffffffffffffULL
50 #else
51 #error unhandled operand size
52 #endif
53
54 /* dynamic flags computation */
55
56 static int glue(compute_all_add, SUFFIX)(void)
57 {
58 int cf, pf, af, zf, sf, of;
59 target_long src1, src2;
60 src1 = CC_SRC;
61 src2 = CC_DST - CC_SRC;
62 cf = (DATA_TYPE)CC_DST < (DATA_TYPE)src1;
63 pf = parity_table[(uint8_t)CC_DST];
64 af = (CC_DST ^ src1 ^ src2) & 0x10;
65 zf = ((DATA_TYPE)CC_DST == 0) << 6;
66 sf = lshift(CC_DST, 8 - DATA_BITS) & 0x80;
67 of = lshift((src1 ^ src2 ^ -1) & (src1 ^ CC_DST), 12 - DATA_BITS) & CC_O;
68 return cf | pf | af | zf | sf | of;
69 }
70
71 static int glue(compute_c_add, SUFFIX)(void)
72 {
73 int cf;
74 target_long src1;
75 src1 = CC_SRC;
76 cf = (DATA_TYPE)CC_DST < (DATA_TYPE)src1;
77 return cf;
78 }
79
80 static int glue(compute_all_adc, SUFFIX)(void)
81 {
82 int cf, pf, af, zf, sf, of;
83 target_long src1, src2;
84 src1 = CC_SRC;
85 src2 = CC_DST - CC_SRC - 1;
86 cf = (DATA_TYPE)CC_DST <= (DATA_TYPE)src1;
87 pf = parity_table[(uint8_t)CC_DST];
88 af = (CC_DST ^ src1 ^ src2) & 0x10;
89 zf = ((DATA_TYPE)CC_DST == 0) << 6;
90 sf = lshift(CC_DST, 8 - DATA_BITS) & 0x80;
91 of = lshift((src1 ^ src2 ^ -1) & (src1 ^ CC_DST), 12 - DATA_BITS) & CC_O;
92 return cf | pf | af | zf | sf | of;
93 }
94
95 static int glue(compute_c_adc, SUFFIX)(void)
96 {
97 int cf;
98 target_long src1;
99 src1 = CC_SRC;
100 cf = (DATA_TYPE)CC_DST <= (DATA_TYPE)src1;
101 return cf;
102 }
103
104 static int glue(compute_all_sub, SUFFIX)(void)
105 {
106 int cf, pf, af, zf, sf, of;
107 target_long src1, src2;
108 src1 = CC_DST + CC_SRC;
109 src2 = CC_SRC;
110 cf = (DATA_TYPE)src1 < (DATA_TYPE)src2;
111 pf = parity_table[(uint8_t)CC_DST];
112 af = (CC_DST ^ src1 ^ src2) & 0x10;
113 zf = ((DATA_TYPE)CC_DST == 0) << 6;
114 sf = lshift(CC_DST, 8 - DATA_BITS) & 0x80;
115 of = lshift((src1 ^ src2) & (src1 ^ CC_DST), 12 - DATA_BITS) & CC_O;
116 return cf | pf | af | zf | sf | of;
117 }
118
119 static int glue(compute_c_sub, SUFFIX)(void)
120 {
121 int cf;
122 target_long src1, src2;
123 src1 = CC_DST + CC_SRC;
124 src2 = CC_SRC;
125 cf = (DATA_TYPE)src1 < (DATA_TYPE)src2;
126 return cf;
127 }
128
129 static int glue(compute_all_sbb, SUFFIX)(void)
130 {
131 int cf, pf, af, zf, sf, of;
132 target_long src1, src2;
133 src1 = CC_DST + CC_SRC + 1;
134 src2 = CC_SRC;
135 cf = (DATA_TYPE)src1 <= (DATA_TYPE)src2;
136 pf = parity_table[(uint8_t)CC_DST];
137 af = (CC_DST ^ src1 ^ src2) & 0x10;
138 zf = ((DATA_TYPE)CC_DST == 0) << 6;
139 sf = lshift(CC_DST, 8 - DATA_BITS) & 0x80;
140 of = lshift((src1 ^ src2) & (src1 ^ CC_DST), 12 - DATA_BITS) & CC_O;
141 return cf | pf | af | zf | sf | of;
142 }
143
144 static int glue(compute_c_sbb, SUFFIX)(void)
145 {
146 int cf;
147 target_long src1, src2;
148 src1 = CC_DST + CC_SRC + 1;
149 src2 = CC_SRC;
150 cf = (DATA_TYPE)src1 <= (DATA_TYPE)src2;
151 return cf;
152 }
153
154 static int glue(compute_all_logic, SUFFIX)(void)
155 {
156 int cf, pf, af, zf, sf, of;
157 cf = 0;
158 pf = parity_table[(uint8_t)CC_DST];
159 af = 0;
160 zf = ((DATA_TYPE)CC_DST == 0) << 6;
161 sf = lshift(CC_DST, 8 - DATA_BITS) & 0x80;
162 of = 0;
163 return cf | pf | af | zf | sf | of;
164 }
165
166 static int glue(compute_c_logic, SUFFIX)(void)
167 {
168 return 0;
169 }
170
171 static int glue(compute_all_inc, SUFFIX)(void)
172 {
173 int cf, pf, af, zf, sf, of;
174 target_long src1, src2;
175 src1 = CC_DST - 1;
176 src2 = 1;
177 cf = CC_SRC;
178 pf = parity_table[(uint8_t)CC_DST];
179 af = (CC_DST ^ src1 ^ src2) & 0x10;
180 zf = ((DATA_TYPE)CC_DST == 0) << 6;
181 sf = lshift(CC_DST, 8 - DATA_BITS) & 0x80;
182 of = ((CC_DST & DATA_MASK) == SIGN_MASK) << 11;
183 return cf | pf | af | zf | sf | of;
184 }
185
186 #if DATA_BITS == 32
187 static int glue(compute_c_inc, SUFFIX)(void)
188 {
189 return CC_SRC;
190 }
191 #endif
192
193 static int glue(compute_all_dec, SUFFIX)(void)
194 {
195 int cf, pf, af, zf, sf, of;
196 target_long src1, src2;
197 src1 = CC_DST + 1;
198 src2 = 1;
199 cf = CC_SRC;
200 pf = parity_table[(uint8_t)CC_DST];
201 af = (CC_DST ^ src1 ^ src2) & 0x10;
202 zf = ((DATA_TYPE)CC_DST == 0) << 6;
203 sf = lshift(CC_DST, 8 - DATA_BITS) & 0x80;
204 of = ((CC_DST & DATA_MASK) == ((target_ulong)SIGN_MASK - 1)) << 11;
205 return cf | pf | af | zf | sf | of;
206 }
207
208 static int glue(compute_all_shl, SUFFIX)(void)
209 {
210 int cf, pf, af, zf, sf, of;
211 cf = (CC_SRC >> (DATA_BITS - 1)) & CC_C;
212 pf = parity_table[(uint8_t)CC_DST];
213 af = 0; /* undefined */
214 zf = ((DATA_TYPE)CC_DST == 0) << 6;
215 sf = lshift(CC_DST, 8 - DATA_BITS) & 0x80;
216 /* of is defined if shift count == 1 */
217 of = lshift(CC_SRC ^ CC_DST, 12 - DATA_BITS) & CC_O;
218 return cf | pf | af | zf | sf | of;
219 }
220
221 static int glue(compute_c_shl, SUFFIX)(void)
222 {
223 return (CC_SRC >> (DATA_BITS - 1)) & CC_C;
224 }
225
226 #if DATA_BITS == 32
227 static int glue(compute_c_sar, SUFFIX)(void)
228 {
229 return CC_SRC & 1;
230 }
231 #endif
232
233 static int glue(compute_all_sar, SUFFIX)(void)
234 {
235 int cf, pf, af, zf, sf, of;
236 cf = CC_SRC & 1;
237 pf = parity_table[(uint8_t)CC_DST];
238 af = 0; /* undefined */
239 zf = ((DATA_TYPE)CC_DST == 0) << 6;
240 sf = lshift(CC_DST, 8 - DATA_BITS) & 0x80;
241 /* of is defined if shift count == 1 */
242 of = lshift(CC_SRC ^ CC_DST, 12 - DATA_BITS) & CC_O;
243 return cf | pf | af | zf | sf | of;
244 }
245
246 #if DATA_BITS == 32
247 static int glue(compute_c_mul, SUFFIX)(void)
248 {
249 int cf;
250 cf = (CC_SRC != 0);
251 return cf;
252 }
253 #endif
254
255 /* NOTE: we compute the flags like the P4. On olders CPUs, only OF and
256 CF are modified and it is slower to do that. */
257 static int glue(compute_all_mul, SUFFIX)(void)
258 {
259 int cf, pf, af, zf, sf, of;
260 cf = (CC_SRC != 0);
261 pf = parity_table[(uint8_t)CC_DST];
262 af = 0; /* undefined */
263 zf = ((DATA_TYPE)CC_DST == 0) << 6;
264 sf = lshift(CC_DST, 8 - DATA_BITS) & 0x80;
265 of = cf << 11;
266 return cf | pf | af | zf | sf | of;
267 }
268
269 /* various optimized jumps cases */
270
271 void OPPROTO glue(op_jb_sub, SUFFIX)(void)
272 {
273 target_long src1, src2;
274 src1 = CC_DST + CC_SRC;
275 src2 = CC_SRC;
276
277 if ((DATA_TYPE)src1 < (DATA_TYPE)src2)
278 GOTO_LABEL_PARAM(1);
279 FORCE_RET();
280 }
281
282 void OPPROTO glue(op_jz_sub, SUFFIX)(void)
283 {
284 if ((DATA_TYPE)CC_DST == 0)
285 GOTO_LABEL_PARAM(1);
286 FORCE_RET();
287 }
288
289 void OPPROTO glue(op_jnz_sub, SUFFIX)(void)
290 {
291 if ((DATA_TYPE)CC_DST != 0)
292 GOTO_LABEL_PARAM(1);
293 FORCE_RET();
294 }
295
296 void OPPROTO glue(op_jbe_sub, SUFFIX)(void)
297 {
298 target_long src1, src2;
299 src1 = CC_DST + CC_SRC;
300 src2 = CC_SRC;
301
302 if ((DATA_TYPE)src1 <= (DATA_TYPE)src2)
303 GOTO_LABEL_PARAM(1);
304 FORCE_RET();
305 }
306
307 void OPPROTO glue(op_js_sub, SUFFIX)(void)
308 {
309 if (CC_DST & SIGN_MASK)
310 GOTO_LABEL_PARAM(1);
311 FORCE_RET();
312 }
313
314 void OPPROTO glue(op_jl_sub, SUFFIX)(void)
315 {
316 target_long src1, src2;
317 src1 = CC_DST + CC_SRC;
318 src2 = CC_SRC;
319
320 if ((DATA_STYPE)src1 < (DATA_STYPE)src2)
321 GOTO_LABEL_PARAM(1);
322 FORCE_RET();
323 }
324
325 void OPPROTO glue(op_jle_sub, SUFFIX)(void)
326 {
327 target_long src1, src2;
328 src1 = CC_DST + CC_SRC;
329 src2 = CC_SRC;
330
331 if ((DATA_STYPE)src1 <= (DATA_STYPE)src2)
332 GOTO_LABEL_PARAM(1);
333 FORCE_RET();
334 }
335
336 /* oldies */
337
338 #if DATA_BITS >= 16
339
340 void OPPROTO glue(op_loopnz, SUFFIX)(void)
341 {
342 if ((DATA_TYPE)ECX != 0 && !(T0 & CC_Z))
343 GOTO_LABEL_PARAM(1);
344 FORCE_RET();
345 }
346
347 void OPPROTO glue(op_loopz, SUFFIX)(void)
348 {
349 if ((DATA_TYPE)ECX != 0 && (T0 & CC_Z))
350 GOTO_LABEL_PARAM(1);
351 FORCE_RET();
352 }
353
354 void OPPROTO glue(op_jz_ecx, SUFFIX)(void)
355 {
356 if ((DATA_TYPE)ECX == 0)
357 GOTO_LABEL_PARAM(1);
358 FORCE_RET();
359 }
360
361 void OPPROTO glue(op_jnz_ecx, SUFFIX)(void)
362 {
363 if ((DATA_TYPE)ECX != 0)
364 GOTO_LABEL_PARAM(1);
365 FORCE_RET();
366 }
367
368 #endif
369
370 /* various optimized set cases */
371
372 void OPPROTO glue(op_setb_T0_sub, SUFFIX)(void)
373 {
374 target_long src1, src2;
375 src1 = CC_DST + CC_SRC;
376 src2 = CC_SRC;
377
378 T0 = ((DATA_TYPE)src1 < (DATA_TYPE)src2);
379 }
380
381 void OPPROTO glue(op_setz_T0_sub, SUFFIX)(void)
382 {
383 T0 = ((DATA_TYPE)CC_DST == 0);
384 }
385
386 void OPPROTO glue(op_setbe_T0_sub, SUFFIX)(void)
387 {
388 target_long src1, src2;
389 src1 = CC_DST + CC_SRC;
390 src2 = CC_SRC;
391
392 T0 = ((DATA_TYPE)src1 <= (DATA_TYPE)src2);
393 }
394
395 void OPPROTO glue(op_sets_T0_sub, SUFFIX)(void)
396 {
397 T0 = lshift(CC_DST, -(DATA_BITS - 1)) & 1;
398 }
399
400 void OPPROTO glue(op_setl_T0_sub, SUFFIX)(void)
401 {
402 target_long src1, src2;
403 src1 = CC_DST + CC_SRC;
404 src2 = CC_SRC;
405
406 T0 = ((DATA_STYPE)src1 < (DATA_STYPE)src2);
407 }
408
409 void OPPROTO glue(op_setle_T0_sub, SUFFIX)(void)
410 {
411 target_long src1, src2;
412 src1 = CC_DST + CC_SRC;
413 src2 = CC_SRC;
414
415 T0 = ((DATA_STYPE)src1 <= (DATA_STYPE)src2);
416 }
417
418 /* shifts */
419
420 void OPPROTO glue(glue(op_shl, SUFFIX), _T0_T1)(void)
421 {
422 int count;
423 count = T1 & SHIFT1_MASK;
424 T0 = T0 << count;
425 FORCE_RET();
426 }
427
428 void OPPROTO glue(glue(op_shr, SUFFIX), _T0_T1)(void)
429 {
430 int count;
431 count = T1 & SHIFT1_MASK;
432 T0 &= DATA_MASK;
433 T0 = T0 >> count;
434 FORCE_RET();
435 }
436
437 void OPPROTO glue(glue(op_sar, SUFFIX), _T0_T1)(void)
438 {
439 int count;
440 target_long src;
441
442 count = T1 & SHIFT1_MASK;
443 src = (DATA_STYPE)T0;
444 T0 = src >> count;
445 FORCE_RET();
446 }
447
448 #undef MEM_WRITE
449 #include "ops_template_mem.h"
450
451 #define MEM_WRITE 0
452 #include "ops_template_mem.h"
453
454 #if !defined(CONFIG_USER_ONLY)
455 #define MEM_WRITE 1
456 #include "ops_template_mem.h"
457
458 #define MEM_WRITE 2
459 #include "ops_template_mem.h"
460 #endif
461
462 /* bit operations */
463 #if DATA_BITS >= 16
464
465 void OPPROTO glue(glue(op_bt, SUFFIX), _T0_T1_cc)(void)
466 {
467 int count;
468 count = T1 & SHIFT_MASK;
469 CC_SRC = T0 >> count;
470 }
471
472 void OPPROTO glue(glue(op_bts, SUFFIX), _T0_T1_cc)(void)
473 {
474 int count;
475 count = T1 & SHIFT_MASK;
476 T1 = T0 >> count;
477 T0 |= (((target_long)1) << count);
478 }
479
480 void OPPROTO glue(glue(op_btr, SUFFIX), _T0_T1_cc)(void)
481 {
482 int count;
483 count = T1 & SHIFT_MASK;
484 T1 = T0 >> count;
485 T0 &= ~(((target_long)1) << count);
486 }
487
488 void OPPROTO glue(glue(op_btc, SUFFIX), _T0_T1_cc)(void)
489 {
490 int count;
491 count = T1 & SHIFT_MASK;
492 T1 = T0 >> count;
493 T0 ^= (((target_long)1) << count);
494 }
495
496 void OPPROTO glue(glue(op_add_bit, SUFFIX), _A0_T1)(void)
497 {
498 A0 += ((DATA_STYPE)T1 >> (3 + SHIFT)) << SHIFT;
499 }
500
501 void OPPROTO glue(glue(op_bsf, SUFFIX), _T0_cc)(void)
502 {
503 int count;
504 target_long res;
505
506 res = T0 & DATA_MASK;
507 if (res != 0) {
508 count = 0;
509 while ((res & 1) == 0) {
510 count++;
511 res >>= 1;
512 }
513 T1 = count;
514 CC_DST = 1; /* ZF = 0 */
515 } else {
516 CC_DST = 0; /* ZF = 1 */
517 }
518 FORCE_RET();
519 }
520
521 void OPPROTO glue(glue(op_bsr, SUFFIX), _T0_cc)(void)
522 {
523 int count;
524 target_long res;
525
526 res = T0 & DATA_MASK;
527 if (res != 0) {
528 count = DATA_BITS - 1;
529 while ((res & SIGN_MASK) == 0) {
530 count--;
531 res <<= 1;
532 }
533 T1 = count;
534 CC_DST = 1; /* ZF = 0 */
535 } else {
536 CC_DST = 0; /* ZF = 1 */
537 }
538 FORCE_RET();
539 }
540
541 #endif
542
543 #if DATA_BITS == 32
544 void OPPROTO op_update_bt_cc(void)
545 {
546 CC_SRC = T1;
547 }
548 #endif
549
550 /* string operations */
551
552 void OPPROTO glue(op_movl_T0_Dshift, SUFFIX)(void)
553 {
554 T0 = DF << SHIFT;
555 }
556
557 /* port I/O */
558 #if DATA_BITS <= 32
559 void OPPROTO glue(glue(op_out, SUFFIX), _T0_T1)(void)
560 {
561 glue(cpu_out, SUFFIX)(env, T0, T1 & DATA_MASK);
562 }
563
564 void OPPROTO glue(glue(op_in, SUFFIX), _T0_T1)(void)
565 {
566 T1 = glue(cpu_in, SUFFIX)(env, T0);
567 }
568
569 void OPPROTO glue(glue(op_in, SUFFIX), _DX_T0)(void)
570 {
571 T0 = glue(cpu_in, SUFFIX)(env, EDX & 0xffff);
572 }
573
574 void OPPROTO glue(glue(op_out, SUFFIX), _DX_T0)(void)
575 {
576 glue(cpu_out, SUFFIX)(env, EDX & 0xffff, T0);
577 }
578
579 void OPPROTO glue(glue(op_check_io, SUFFIX), _T0)(void)
580 {
581 glue(glue(check_io, SUFFIX), _T0)();
582 }
583
584 void OPPROTO glue(glue(op_check_io, SUFFIX), _DX)(void)
585 {
586 glue(glue(check_io, SUFFIX), _DX)();
587 }
588 #endif
589
590 #undef DATA_BITS
591 #undef SHIFT_MASK
592 #undef SHIFT1_MASK
593 #undef SIGN_MASK
594 #undef DATA_TYPE
595 #undef DATA_STYPE
596 #undef DATA_MASK
597 #undef SUFFIX