moo/moo/kernel/Magnitu.moo
hyunghwan.chung c85bf336ad changed the error literal notation from error(NNN) to #\NNN.
prohibited the backslash character from forming a binary selector.
renamed quo: to div:, // to mdiv:, \ to mod:
2018-04-08 17:47:38 +00:00

385 lines
5.7 KiB
Smalltalk

class Magnitude(Object)
{
method < aMagnitude { self subclassResponsibility: #< }
method > aMagnitude { ^aMagnitude < self }
method <= aMagnitude { ^(aMagnitude < self) not }
method >= aMagnitude { ^(self < aMagnitude) not }
method between: min and: max
{
^self >= min and: [self <= max]
}
method min: aMagnitude
{
^self < aMagnitude ifTrue: [self] ifFalse: [aMagnitude]
}
method max: aMagnitude
{
^self > aMagnitude ifTrue: [self] ifFalse: [aMagnitude]
}
}
class Association(Magnitude)
{
var key, value.
method(#class) key: key value: value
{
^self new key: key value: value
}
method key: key value: value
{
self.key := key.
self.value := value.
}
method value: value
{
self.value := value
}
method key
{
^self.key
}
method value
{
^self.value
}
method = ass
{
^(self.key = ass key) and: [ self.value = ass value ]
}
method hash
{
^(self.key hash) + (self.value hash)
}
}
class(#limited) Character(Magnitude)
{
## method(#primitive,#class) codePoint: code.
## method(#primitive) codePoint.
method asCharacter { ^self }
method(#primitive) asError.
method(#primitive) asInteger.
method(#primitive) < char.
method(#primitive) > char.
method(#primitive) <= char.
method(#primitive) >= char.
method digitValue: anInteger
{
^'0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ' at: anInteger
}
method digitValue
{
##<primitive: #Character_digitValue>
if (self >= $0 and: [self <= $9])
{
^self asInteger - $0 asInteger
}
elsif (self >= $A and: [self <= $Z])
{
^self asInteger - $A asInteger + 10
}
elsif (self >= $a and: [self <= $z])
{
^self asInteger - $a asInteger + 10
}.
##Exception signal: 'not a digit character'.
^-1
}
}
class(#limited) Number(Magnitude)
{
method + aNumber
{
<primitive: #_integer_add>
self primitiveFailed.
}
method - aNumber
{
<primitive: #_integer_sub>
self primitiveFailed.
}
method * aNumber
{
<primitive: #_integer_mul>
self primitiveFailed.
}
method div: aNumber
{
## integer division rounded toward zero
<primitive: #_integer_div>
self primitiveFailed.
}
method rem: aNumber
{
## integer remainder rounded toward zero
<primitive: #_integer_rem>
self primitiveFailed.
}
method mdiv: aNumber
{
## integer division quotient
<primitive: #_integer_mdiv>
self primitiveFailed.
}
method mod: aNumber
{
## integer division remainder
<primitive: #_integer_mod>
self primitiveFailed.
}
##method / aNumber
##{
## ## fraction? fixed-point decimal? floating-point?
##}
method = aNumber
{
<primitive: #_integer_eq>
self primitiveFailed.
}
method ~= aNumber
{
<primitive: #_integer_ne>
self primitiveFailed.
}
method < aNumber
{
<primitive: #_integer_lt>
self primitiveFailed.
}
method > aNumber
{
<primitive: #_integer_gt>
self primitiveFailed.
}
method <= aNumber
{
<primitive: #_integer_le>
self primitiveFailed.
}
method >= aNumber
{
<primitive: #_integer_ge>
self primitiveFailed.
}
method negated
{
<primitive: #_integer_negated>
^0 - self.
}
method bitAt: index
{
<primitive: #_integer_bitat>
^(self bitShift: index negated) bitAnd: 1.
}
method bitAnd: aNumber
{
<primitive: #_integer_bitand>
self primitiveFailed.
}
method bitOr: aNumber
{
<primitive: #_integer_bitor>
self primitiveFailed.
}
method bitXor: aNumber
{
<primitive: #_integer_bitxor>
self primitiveFailed.
}
method bitInvert
{
<primitive: #_integer_bitinv>
^-1 - self.
}
method bitShift: aNumber
{
(* positive number for left shift.
* negative number for right shift *)
<primitive: #_integer_bitshift>
self primitiveFailed.
}
method asString
{
^self printStringRadix: 10
}
method printStringRadix: aNumber
{
<primitive: #_integer_inttostr>
self primitiveFailed.
}
method to: end by: step do: block
{
| i |
i := self.
(*
(step > 0)
ifTrue: [
[ i <= end ] whileTrue: [
block value: i.
i := i + step.
].
]
ifFalse: [
[ i >= end ] whileTrue: [
block value: i.
i := i - step.
].
].
*)
if (step > 0)
{
while (i <= end)
{
block value: i.
i := i + step.
}
}
else
{
while ( i => end)
{
block value: i.
i := i - step.
}
}
}
method to: end do: block
{
^self to: end by: 1 do: block.
}
method priorTo: end by: step do: block
{
| i |
i := self.
(*
(step > 0)
ifTrue: [
[ i < end ] whileTrue: [
block value: i.
i := i + step.
].
]
ifFalse: [
[ i > end ] whileTrue: [
block value: i.
i := i - step.
].
].
*)
if (step > 0)
{
while (i < end)
{
block value: i.
i := i + step.
}
}
else
{
while ( i > end)
{
block value: i.
i := i - step.
}
}
}
method priorTo: end do: block
{
^self priorTo: end by: 1 do: block.
}
method abs
{
(*self < 0 ifTrue: [^self negated].
^self.*)
^if (self < 0) { self negated } else { self }
}
method sign
{
(* self < 0 ifTrue: [^-1].
self > 0 ifTrue: [^1].
^0.*)
^if (self < 0) { -1 } elsif (self > 0) { 1 } else { 0 }
}
}
class(#limited) Integer(Number)
{
method timesRepeat: aBlock
{
## 1 to: self by: 1 do: [ :count | aBlock value ].
| count |
count := 0.
while (count < self) { aBlock value. count := count + 1 }
}
method asInteger { ^self }
}
class(#limited) SmallInteger(Integer)
{
method(#primitive) asCharacter.
method(#primitive) asError.
}
class(#liword,#limited) LargeInteger(Integer)
{
}
class(#liword,#immutable) LargePositiveInteger(LargeInteger)
{
method abs { ^self }
method sign { ^1 }
}
class(#liword,#immutable) LargeNegativeInteger(LargeInteger)
{
method abs { ^self negated }
method sign { ^-1 }
}