File: interp.inl

package info (click to toggle)
gnu-smalltalk 3.0.3-2
  • links: PTS
  • area: main
  • in suites: lenny
  • size: 26,460 kB
  • ctags: 13,326
  • sloc: ansic: 82,059; sh: 22,788; asm: 7,846; perl: 4,493; cpp: 3,517; awk: 1,471; yacc: 1,355; makefile: 1,151; xml: 886; lex: 843; sed: 258
file content (96 lines) | stat: -rw-r--r-- 3,742 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
/******************************** -*- C -*- ****************************
 *
 *	Byte code interpreter inlines
 *
 *
 ***********************************************************************/

/***********************************************************************
 *
 * Copyright 2000, 2001, 2002, 2006 Free Software Foundation, Inc.
 * Written by Steve Byrne.
 *
 * This file is part of GNU Smalltalk.
 *
 * GNU Smalltalk is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the Free
 * Software Foundation; either version 2, or (at your option) any later 
 * version.
 * 
 * Linking GNU Smalltalk statically or dynamically with other modules is
 * making a combined work based on GNU Smalltalk.  Thus, the terms and
 * conditions of the GNU General Public License cover the whole
 * combination.
 *
 * In addition, as a special exception, the Free Software Foundation
 * give you permission to combine GNU Smalltalk with free software
 * programs or libraries that are released under the GNU LGPL and with
 * independent programs running under the GNU Smalltalk virtual machine.
 *
 * You may copy and distribute such a system following the terms of the
 * GNU GPL for GNU Smalltalk and the licenses of the other code
 * concerned, provided that you include the source code of that other
 * code when and as the GNU GPL requires distribution of source code.
 *
 * Note that people who make modified versions of GNU Smalltalk are not
 * obligated to grant this special exception for their modified
 * versions; it is their choice whether to do so.  The GNU General
 * Public License gives permission to release a modified version without
 * this exception; this exception also makes it possible to release a
 * modified version which carries forward this exception.
 *
 * GNU Smalltalk is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 * more details.
 * 
 * You should have received a copy of the GNU General Public License along with
 * GNU Smalltalk; see the file COPYING.  If not, write to the Free Software
 * Foundation, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.  
 *
 ***********************************************************************/


/* Multiply a*b and in case of an overflow, answer OVERFLOWING_INT so
   that we can work it out the same way we do with adds and
   subtracts.  */
static inline intptr_t mul_with_check (intptr_t a, intptr_t b);

/* using STACK_AT is correct: numArgs == 0 means that there's just the
 * receiver on the stack, at 0.  numArgs = 1 means that at location 0 is
 * the arg, location 1 is the receiver.  And so on.  */
#define SEND_MESSAGE(sendSelector, sendArgs) do {		\
  OOP _receiver;						\
  _receiver = STACK_AT(sendArgs);				\
  _gst_send_message_internal(sendSelector, sendArgs, _receiver,	\
			     OOP_INT_CLASS(_receiver));		\
} while(0)


intptr_t
mul_with_check (intptr_t a, intptr_t b)
{
  intmax_t result = (intmax_t)a * b;

  /* We define the largest int type in stdintx.h, but we can
     only use it if it is two times the width of an intptr_t.  */

  if (sizeof (intmax_t) >= 2 * sizeof (intptr_t))
    {
      if UNCOMMON (result > MAX_ST_INT || result < MIN_ST_INT)
        return (OVERFLOWING_INT);
      else
        return (result);
    }

  /* This fallback method uses a division to do overflow check */
  else
    {
      if COMMON (((uintptr_t) (a | b)) < (1L << (ST_INT_SIZE / 2))
	         || b == 0
	         || result / b == a)
        return (result);
      else
        return (OVERFLOWING_INT);
    }
}