File: IntInf.sml

package info (click to toggle)
polyml 5.6-8
  • links: PTS, VCS
  • area: main
  • in suites: stretch
  • size: 31,892 kB
  • ctags: 34,453
  • sloc: cpp: 44,983; ansic: 24,520; asm: 14,850; sh: 11,730; makefile: 551; exp: 484; python: 253; awk: 91; sed: 9
file content (128 lines) | stat: -rw-r--r-- 4,476 bytes parent folder | download
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
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
(*
    Title:      Standard Basis Library: IntInf structure and signature.
    Author:     David Matthews
    Copyright   David Matthews 2000

    This library is free software; you can redistribute it and/or
    modify it under the terms of the GNU Lesser General Public
    License as published by the Free Software Foundation; either
    version 2.1 of the License, or (at your option) any later version.
    
    This library 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
    Lesser General Public License for more details.
    
    You should have received a copy of the GNU Lesser General Public
    License along with this library; if not, write to the Free Software
    Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
*)

(* G&R 2004 status: checked, no change. *)

signature INT_INF =
sig
    include INTEGER
    val divMod : int * int -> int * int
    val quotRem : int * int -> int * int
    val pow : int * Int.int -> int
    val log2 : int -> Int.int
    val orb : int * int -> int
    val xorb : int * int -> int
    val andb : int * int -> int
    val notb : int -> int
    val << : int * Word.word -> int
    val ~>> : int * Word.word -> int
end;

structure IntInf : INT_INF =
struct
    open RuntimeCalls;
    
    val quotRem: int*int->int*int = RunCall.run_call2C2 POLY_SYS_quotrem

    (* This should really be defined in terms of quotRem. *)
    fun divMod(i, j) = (i div j, i mod j)

    (* Return the position of the highest bit set in the value. *)
    local
        val isShort: int -> bool = RunCall.run_call1 POLY_SYS_is_short
        val loadByte: int*int->word = RunCall.run_call2 POLY_SYS_load_byte
        val segLength: int -> int = RunCall.run_call1 POLY_SYS_get_length

        (* Compute log2 for a short value.  The top bit of i will always be
           zero since we've checked that it's positive so it will always
           terminate. *)
        fun log2Word(i: word, j: word, n: int) =
            if Word.>(j, i) then n-1
            else log2Word(i, Word.<<(j, 0w1), n+1)

        (* The value is represented as little-endian byte segment.
           High-order bytes may be zero so we work back until we
           find a non-zero byte and then find the bit-position
           within it. *)
        fun log2Long(i, byte) =
        let
            val b = loadByte(i, byte)
        in
            if b = 0w0 then log2Long(i, byte-1)
            else log2Word(b, 0w2, 1) + byte*8
        end
            
    in
        fun log2 i =
            if i <= 0 then raise Domain
            else if isShort i
            then log2Word(Word.fromInt i, 0w2, 1)
            else (* i is actually a pointer to a byte segment. *)
            let
                val bytes =
                    segLength i * RunCall.unsafeCast LibrarySupport.wordSize
            in
               log2Long(i, bytes-1)
            end
    end

    (* These are implemented in the RTS. *)
    val orb : int * int -> int = RunCall.run_call2 POLY_SYS_ora
    val xorb : int * int -> int = RunCall.run_call2 POLY_SYS_xora
    val andb : int * int -> int = RunCall.run_call2 POLY_SYS_anda

    (* notb is defined as ~ (i+1) and there doesn't seem to be much advantage
       in implementing it any other way. *)
    fun notb i = ~(i + 1)

    fun pow(i, j) =
    let
        fun power(acc, _, 0) = acc
         |  power(acc, n, i) =
            power(if andb(i, 1) = 1 then acc*n else acc, n*n, Int.quot(i, 2))
    in
        if j < 0
        then (* Various exceptional cases. *)
            (
            if i = 0 then raise Div
            else if i = 1 then 1
            else if i = ~1
            then if andb(j, 1) = 0 then (*even*) 1 else (*odd*) ~1
            else 0
            )
        else power(1, i, j)
    end

    (* These could be implemented in the RTS although I doubt if it's
       really worth it. *)
    local
        val maxShift = Word.fromInt Word.wordSize
        val fullShift = pow(2, Word.wordSize)
    in
        fun << (i: int, j: Word.word) =
           if j < maxShift
           then i * Word.toInt(Word.<<(0w1, j))
           else <<(i * fullShift, j-maxShift)
    end
    
    fun ~>> (i: int, j: Word.word) = Int.div(i, pow(2, Word.toInt j))

    open Int (* Inherit everything from Int.  Do this last because it overrides the overloaded functions. *)
end;