MIXAL
memory.cpp
1 #include <stdexcept>
2 #include <string>
3 #include <iostream>
4 #include "memory.h"
5 #include "unicode_char.h"
6 
7 namespace mixal {
8 
9 uint16_t CHAR_CODES[] = {
10  ' ', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I',
11  0xb4, 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R',
12  0x2da, 0x2dd, 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z',
13  '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
14  '.', ',', '(', ')', '+', '-', '*', '/', '=', '$',
15  '<', '>', '@', ';', ':', 0x201a
16 };
17 
18 bool ComputerWord::operator==(const ComputerWord& word) const {
19  return negative == word.negative && byte1 == word.byte1 &&
20  byte2 == word.byte2 && byte3 == word.byte3 &&
21  byte4 == word.byte4 && byte5 == word.byte5;
22 }
23 
24 std::ostream& operator<<(std::ostream& out, const ComputerWord& word) {
25  out << (word.negative ? '-' : '+');
26  for (int i = 1; i <= 5; ++i) {
27  out << ' ' << static_cast<int>(word[i]);
28  }
29  return out;
30 }
31 
32 ComputerWord::ComputerWord() : negative(), byte1(), byte2(), byte3(), byte4(), byte5() {
33 }
34 
35 ComputerWord::ComputerWord(int32_t value) : negative(), byte1(), byte2(), byte3(), byte4(), byte5() {
36  set(value);
37 }
38 
39 ComputerWord::ComputerWord(const std::string& chars) : negative(), byte1(), byte2(), byte3(), byte4(), byte5() {
40  set(chars);
41 }
42 
44  uint8_t _byte1, uint8_t _byte2, uint8_t _byte3, uint8_t _byte4, uint8_t _byte5) :
45  negative(_negative), byte1(_byte1), byte2(_byte2), byte3(_byte3), byte4(_byte4), byte5(_byte5) {
46 }
47 
48 ComputerWord::ComputerWord(char sign, uint8_t _byte1, uint8_t _byte2, uint8_t _byte3, uint8_t _byte4, uint8_t _byte5) :
49  negative(sign == '-'), byte1(_byte1), byte2(_byte2), byte3(_byte3), byte4(_byte4), byte5(_byte5) {
50  if (sign != '+' && sign != '-') {
51  throw std::runtime_error("Invalid sign: " + std::string(1, sign));
52  }
53 }
54 
55 ComputerWord::ComputerWord(bool _negative, uint16_t bytes12, uint8_t _byte3, uint8_t _byte4, uint8_t _byte5) :
56  negative(_negative), byte1(bytes12 / 64), byte2(bytes12 % 64), byte3(_byte3), byte4(_byte4), byte5(_byte5) {
57 }
58 
59 ComputerWord::ComputerWord(char sign, uint16_t bytes12, uint8_t _byte3, uint8_t _byte4, uint8_t _byte5) :
60  negative(sign == '-'), byte1(bytes12 / 64), byte2(bytes12 % 64), byte3(_byte3), byte4(_byte4), byte5(_byte5) {
61  if (sign != '+' && sign != '-') {
62  throw std::runtime_error("Invalid sign: " + std::string(1, sign));
63  }
64 }
65 
67  negative = false;
68  byte1 = byte2 = byte3 = byte4 = byte5 = 0;
69 }
70 
71 std::string ComputerWord::getBytesString() const {
72  std::string result;
73  result += negative ? '-' : '+';
74  for (int i = 1; i <= 5; ++i) {
75  result += ' ';
76  if ((*this)[i] < 10) {
77  result += ' ';
78  }
79  result += std::to_string(static_cast<int>((*this)[i]));
80  }
81  return result;
82 }
83 
84 uint8_t ComputerWord::operator[](int index) const {
85  if (index <= 0 || index > 5) {
86  throw std::runtime_error("Invalid index for a word: " + std::to_string(index));
87  }
88  switch (index) {
89  case 1: return byte1;
90  case 2: return byte2;
91  case 3: return byte3;
92  case 4: return byte4;
93  default: return byte5;
94  }
95 }
96 
97 uint8_t& ComputerWord::operator[](int index) {
98  if (index <= 0 || index > 5) {
99  throw std::runtime_error("Invalid index for a word: " + std::to_string(index));
100  }
101  switch (index) {
102  case 1: return byte1;
103  case 2: return byte2;
104  case 3: return byte3;
105  case 4: return byte4;
106  default: return byte5;
107  }
108 }
109 
110 uint16_t ComputerWord::bytes2(int index1, int index2) const {
111  int16_t high = static_cast<int16_t>(static_cast<uint8_t>((*this)[index1]));
112  int16_t low = static_cast<int16_t>(static_cast<uint8_t>((*this)[index2]));
113  return high * 64 + low;
114 }
115 
116 uint16_t ComputerWord::bytes12() const {
117  return bytes2(1, 2);
118 }
119 
120 uint16_t ComputerWord::bytes23() const {
121  return bytes2(2, 3);
122 }
123 
124 uint16_t ComputerWord::bytes34() const {
125  return bytes2(3, 4);
126 }
127 
128 uint16_t ComputerWord::bytes45() const {
129  return bytes2(4, 5);
130 }
131 
132 int32_t ComputerWord::value() const {
133  int32_t value = static_cast<int32_t>(byte1 << 24) |
134  static_cast<int32_t>(byte2 << 18) |
135  static_cast<int32_t>(byte3 << 12) |
136  static_cast<int32_t>(byte4 << 6) |
137  static_cast<int32_t>(byte5);
138  return negative ? -value : value;
139 }
140 
141 int16_t ComputerWord::addressValue() const {
142  int16_t value = static_cast<int16_t>(this->bytes12());
143  if (negative) {
144  value = -value;
145  }
146  return value;
147 }
148 
149 void ComputerWord::setAddress(int16_t address) {
150  negative = false;
151  if (address < 0) {
152  negative = true;
153  address = -address;
154  }
155  byte1 = static_cast<uint8_t>(address / 64);
156  byte2 = static_cast<uint8_t>(address % 64);
157 }
158 
159 void ComputerWord::setAddress(bool negative, uint16_t address) {
160  this->negative = negative;
161  byte1 = static_cast<uint8_t>(address / 64);
162  byte2 = static_cast<uint8_t>(address % 64);
163 }
164 
165 std::string ComputerWord::getCharacters() const {
166  std::string chars;
167  for (int i = 1; i <= 5; ++i) {
168  if (getAt(i) < CHAR_CODES_NUM) {
169  uint16_t code = CHAR_CODES[getAt(i)];
170  chars += unicode::toUTF8(static_cast<unicode::UChar>(code));
171  } else {
172  chars += ' ';
173  }
174  }
175  return chars;
176 }
177 
178 void ComputerWord::set(int32_t value) {
179  if (value > 0) {
180  negative = false;
181  } else if (value < 0) {
182  negative = true;
183  value = -value;
184  }
185  for (int i = 5; i >= 1; --i) {
186  set(i, static_cast<uint8_t>(value & ((1 << 6) - 1)));
187  value >>= 6;
188  }
189 }
190 
191 void ComputerWord::set(const std::string& chars) {
192  auto codes = unicode::fromUTF8(chars);
193  if (codes.size() != 5) {
194  throw std::runtime_error("Invalid length of characters for a word: " + chars);
195  }
196  negative = false;
197  for (int i = 0; i < 5; ++i) {
198  (*this)[i + 1] = 0;
199  for (int j = 0; j < CHAR_CODES_NUM; ++j) {
200  if (static_cast<uint16_t>(codes[i]) == CHAR_CODES[j]) {
201  (*this)[i + 1] = j;
202  break;
203  }
204  }
205  }
206 }
207 
208 void ComputerWord::set(int index, uint8_t val) {
209  if (index <= 0 || index > 5) {
210  throw std::runtime_error("Invalid index for a word: " + std::to_string(index));
211  }
212  switch (index) {
213  case 1: byte1 = val; break;
214  case 2: byte2 = val; break;
215  case 3: byte3 = val; break;
216  case 4: byte4 = val; break;
217  default: byte5 = val;
218  }
219 }
220 
221 void ComputerWord::set(bool negative, uint8_t byte1, uint8_t byte2, uint8_t byte3, uint8_t byte4, uint8_t byte5) {
222  this->negative = negative;
223  this->byte1 = byte1;
224  this->byte2 = byte2;
225  this->byte3 = byte3;
226  this->byte4 = byte4;
227  this->byte5 = byte5;
228 }
229 
230 void ComputerWord::set(char sign, uint8_t byte1, uint8_t byte2, uint8_t byte3, uint8_t byte4, uint8_t byte5) {
231  if (sign != '+' && sign != '-') {
232  throw std::runtime_error("Invalid sign: " + std::string(1, sign));
233  }
234  this->negative = sign == '-';
235  this->byte1 = byte1;
236  this->byte2 = byte2;
237  this->byte3 = byte3;
238  this->byte4 = byte4;
239  this->byte5 = byte5;
240 }
241 
242 void ComputerWord::set(bool negative, uint16_t bytes12, uint8_t byte3, uint8_t byte4, uint8_t byte5) {
243  this->negative = negative;
244  this->byte1 = static_cast<uint8_t>(bytes12 / 64);
245  this->byte2 = static_cast<uint8_t>(bytes12 % 64);
246  this->byte3 = byte3;
247  this->byte4 = byte4;
248  this->byte5 = byte5;
249 }
250 
251 void ComputerWord::set(char sign, uint16_t bytes12, uint8_t byte3, uint8_t byte4, uint8_t byte5) {
252  if (sign != '+' && sign != '-') {
253  throw std::runtime_error("Invalid sign: " + std::string(1, sign));
254  }
255  this->negative = sign == '-';
256  this->byte1 = static_cast<uint8_t>(bytes12 / 64);
257  this->byte2 = static_cast<uint8_t>(bytes12 % 64);
258  this->byte3 = byte3;
259  this->byte4 = byte4;
260  this->byte5 = byte5;
261 }
262 
263 }; // namespace mixal
mixal::ComputerWord::address
uint16_t address() const
Definition: memory.h:134
mixal::CHAR_CODES_NUM
const int32_t CHAR_CODES_NUM
Definition: memory.h:17
mixal::ComputerWord::bytes23
uint16_t bytes23() const
Definition: memory.cpp:120
mixal::ComputerWord::bytes45
uint16_t bytes45() const
Definition: memory.cpp:128
mixal::ComputerWord
Definition: memory.h:25
mixal::ComputerWord::bytes12
uint16_t bytes12() const
Definition: memory.cpp:116
mixal::ComputerWord::getBytesString
std::string getBytesString() const
Definition: memory.cpp:71
mixal::ComputerWord::bytes34
uint16_t bytes34() const
Definition: memory.cpp:124
mixal::ComputerWord::reset
void reset()
Definition: memory.cpp:66
mixal::ComputerWord::index
uint8_t index() const
Definition: memory.h:140
mixal::ComputerWord::getAt
uint8_t getAt(int32_t index) const
Definition: memory.h:172
mixal::ComputerWord::operator[]
uint8_t operator[](int index) const
Definition: memory.cpp:84
mixal::ComputerWord::setAddress
void setAddress(int16_t address)
Definition: memory.cpp:149
mixal::ComputerWord::addressValue
int16_t addressValue() const
Definition: memory.cpp:141
mixal::ComputerWord::value
int32_t value() const
Definition: memory.cpp:132
mixal::ComputerWord::set
void set(int32_t value)
Definition: memory.cpp:178
mixal::CHAR_CODES
uint16_t CHAR_CODES[]
Definition: memory.cpp:9
mixal::ComputerWord::getCharacters
std::string getCharacters() const
Definition: memory.cpp:165
mixal::ComputerWord::operator==
bool operator==(const ComputerWord &word) const
Definition: memory.cpp:18
mixal::ComputerWord::bytes2
uint16_t bytes2(int index1, int index2) const
Definition: memory.cpp:110
mixal::operator<<
std::ostream & operator<<(std::ostream &out, Operation operation)
Definition: expression.cpp:335
mixal::ComputerWord::ComputerWord
ComputerWord()
Definition: memory.cpp:32
memory.h
The definition of a computer word.