File: cell.h

package info (click to toggle)
mixal 1.08-10
  • links: PTS
  • area: main
  • in suites: etch, etch-m68k, lenny, squeeze
  • size: 208 kB
  • ctags: 286
  • sloc: ansic: 1,597; makefile: 89; awk: 10
file content (101 lines) | stat: -rw-r--r-- 3,303 bytes parent folder | download | duplicates (5)
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
/* MIX simulator, copyright 1994 by Darius Bacon */ 
#include "mix.h"

#ifndef CELL_H
#define CELL_H

/* --- Call this before anything else --- */
void precompute_field_data(void);

/*
 * MIX values ---
 *
 * Currently represented as a long.  (A long must have >= 31 bits.)
 * Bit 30 is the sign, higher bits are 0, and bits 0-29 are the magnitude.
 * (If we made the high bit the sign, would it be a little faster to check?)
 * Each MIX 'byte' is a 6-bit substring of the magnitude.
 *
 * Other reps which might make sense:
 *   array of bytes
 *   pair (sign, magnitude)
 *   1's-complement??? 
 *       (a kind of halfway house between 2's-complement and sign/magnitude)
 *       (alas, field operations would be more work...)
 * Unfortunately, in several places I've made the convenient assumption that
 * the magnitude of a MIX cell can be converted to a C long int; that would 
 * be wrong for, e.g., a decimal MIX machine if a long int is only 32 bits.
 */

typedef long Cell;
typedef unsigned Byte;  /* 0..63 */

/* These are to allow symbol.c to convert a Cell to or from a unique string
   representation.  We could just as easily use unparse() if we defined a
   complementary parse() function. */
#define cell_to_long(c) (c)
#define long_to_cell(l) (l)

/* Largest possible magnitude of a Cell */
#define CELL_MAX    ((1L << 30) - 1)
/* Largest possible magnitude of an index register (i.e., 2 bytes) */
#define INDEX_MAX   ((1L << 12) - 1)

#define zero ((Cell) 0)

#define the_sign_bit    (1L << 30)

#define negative(cell)      ( (cell) ^ the_sign_bit )
#define sign_bit(cell)      ( (cell) & the_sign_bit )
#define magnitude(cell)     ( (cell) & (the_sign_bit - 1) )

#define is_negative(cell)   ( sign_bit(cell) != 0 )

Cell ulong_to_cell(unsigned long n);

Byte make_field_spec(unsigned L, unsigned R);
void assert_valid_field(Cell field_spec);

/*** should separate into fast/safe versions */
Cell field(Byte field_spec, Cell cell);
Cell set_field(Cell value, Byte field_spec, Cell into);

Byte get_byte(unsigned field, Cell cell);
Cell set_byte(Byte value, unsigned field, Cell into);

extern Flag overflow;

Cell add(Cell x, Cell y);
Cell sub(Cell x, Cell y);

void multiply(Cell x, Cell y, Cell *high_word, Cell *low_word);
Cell mul(Cell x, Cell y);

void divide(Cell n1, Cell n0, Cell d, Cell *quotient, Cell *remainder);
Cell slash(Cell x, Cell y);     /* the name 'div' is taken... */

void shift_left(Cell A, Cell X, unsigned long count, Cell *pA, Cell *pX);
void shift_right(Cell A, Cell X, unsigned long count, Cell *pA, Cell *pX);
void shift_left_circular(Cell A, Cell X, unsigned count, Cell *pA, Cell *pX);

/* Break a cell into its component fields. */
#define destructure_cell(cell, A, I, F, C)  \
    do {                                    \
	Cell temp_ = (cell);                \
	Cell sign_ = sign_bit(temp_);       \
	C = temp_ & 63;                     \
	F = (temp_ >>= 6) & 63;             \
	I = (temp_ >>= 6) & 63;             \
	A = sign_ | (4095 & (temp_ >> 6));  \
    } while (0)

/* --- Printable representation --- */
void print_cell(Cell cell);
void unparse_cell(char *buffer, Cell cell); /* Pre: 12 <= sizeof(buffer) */

/* --- Addresses --- */
typedef unsigned Address;

Cell address_to_cell(Address addr);
Address cell_to_address(Cell cell);

#endif