File : ffa_calc.adb
1 ------------------------------------------------------------------------------
2 ------------------------------------------------------------------------------
3 -- This file is part of 'Finite Field Arithmetic', aka 'FFA'. --
4 -- --
5 -- (C) 2017 Stanislav Datskovskiy ( www.loper-os.org ) --
6 -- http://wot.deedbot.org/17215D118B7239507FAFED98B98228A001ABFFC7.html --
7 -- --
8 -- You do not have, nor can you ever acquire the right to use, copy or --
9 -- distribute this software ; Should you use this software for any purpose, --
10 -- or copy and distribute it to anyone or in any manner, you are breaking --
11 -- the laws of whatever soi-disant jurisdiction, and you promise to --
12 -- continue doing so for the indefinite future. In any case, please --
13 -- always : read and understand any software ; verify any PGP signatures --
14 -- that you use - for any purpose. --
15 -- --
16 -- See also http://trilema.com/2015/a-new-software-licensing-paradigm . --
17 ------------------------------------------------------------------------------
18 ------------------------------------------------------------------------------
19
20 -- Basics
21 with OS; use OS;
22 with CmdLine; use CmdLine;
23
24 -- FFA
25 with FFA; use FFA;
26
27 -- For the intrinsic equality operator on Words
28 use type FFA.Word;
29
30 -- For RNG:
31 with FFA_RNG; use FFA_RNG;
32
33
34 procedure FFA_Calc is
35
36 Width : Positive; -- Desired FFA Width
37 Height : Positive; -- Desired Height of Stack
38 RNG : RNG_Device; -- The active RNG device.
39
40 begin
41 if Arg_Count < 3 or Arg_Count > 4 then
42 Eggog("Usage: ./ffa_calc WIDTH HEIGHT [/dev/rng]");
43 end if;
44
45 declare
46 Arg1 : CmdLineArg;
47 Arg2 : CmdLineArg;
48 begin
49 -- Get commandline args:
50 Get_Argument(1, Arg1); -- First arg
51 Get_Argument(2, Arg2); -- Second arg
52
53 if Arg_Count = 4 then
54 -- RNG was specified:
55 declare
56 Arg3 : CmdLineArg;
57 begin
58 Get_Argument(3, Arg3); -- Third arg (optional)
59
60 -- Ada.Sequential_IO chokes on paths with trailing whitespace!
61 -- So we have to give it a trimmed path. But we can't use
62 -- Ada.Strings.Fixed.Trim, because it suffers from
63 -- SecondaryStackism-syphilis. Instead we are stuck doing this:
64 Init_RNG(RNG, Arg3(Arg3'First .. Len_Arg(3)));
65 end;
66 else
67 -- RNG was NOT specified:
68 Init_RNG(RNG); -- Use the machine default then
69 end if;
70
71 -- Parse into Positives:
72 Width := Positive'Value(Arg1);
73 Height := Positive'Value(Arg2);
74 exception
75 when others =>
76 Eggog("Invalid arguments!");
77 end;
78
79 -- Test if proposed Width is permissible:
80 if not FFA_FZ_Valid_Bitness_P(Width) then
81 Eggog("Invalid Width: " & FFA_Validity_Rule_Doc);
82 end if;
83
84 -- The Calculator itself:
85 declare
86
87 -- The number of Words required to make a FZ of the given Bitness.
88 Wordness : Indices := Indices(Width / Bitness);
89
90 --------------------------------------------------------
91 -- State --
92 --------------------------------------------------------
93 -- The Stack:
94 subtype Stack_Positions is Natural range 0 .. Height;
95 type Stacks is array(Stack_Positions range <>) of FZ(1 .. Wordness);
96 Stack : Stacks(Stack_Positions'Range);
97
98 -- Stack Pointer:
99 SP : Stack_Positions := Stack_Positions'First;
100
101 -- Carry/Borrow Flag:
102 Flag : WBool := 0;
103
104 -- Odometer:
105 Pos : Natural := 0;
106
107 -- The current levels of the three types of nestedness:
108 QuoteLevel : Natural := 0;
109 CommLevel : Natural := 0;
110 CondLevel : Natural := 0;
111 --------------------------------------------------------
112
113
114 -- Clear the stack and set SP to bottom.
115 procedure Zap is
116 begin
117 -- Clear the stack
118 for i in Stack'Range loop
119 FFA_FZ_Clear(Stack(i));
120 end loop;
121 -- Set SP to bottom
122 SP := Stack_Positions'First;
123 -- Clear Overflow flag
124 Flag := 0;
125 end Zap;
126
127
128 -- Report a fatal error condition at the current symbol
129 procedure E(S : in String) is
130 begin
131 Eggog("Pos:" & Natural'Image(Pos) & ": " & S);
132 end E;
133
134
135 -- Move SP up
136 procedure Push is
137 begin
138 if SP = Stack_Positions'Last then
139 E("Stack Overflow!");
140 else
141 SP := SP + 1;
142 end if;
143 end Push;
144
145
146 -- Discard the top of the stack
147 procedure Drop is
148 begin
149 FFA_FZ_Clear(Stack(SP));
150 SP := SP - 1;
151 end Drop;
152
153
154 -- Check if stack has the necessary N items
155 procedure Want(N : in Positive) is
156 begin
157 if SP < N then
158 E("Stack Underflow!");
159 end if;
160 end Want;
161
162
163 -- Ensure that a divisor is not zero
164 procedure MustNotZero(D : in FZ) is
165 begin
166 if FFA_FZ_ZeroP(D) = 1 then
167 E("Division by Zero!");
168 end if;
169 end MustNotZero;
170
171
172 -- Slide a new hex digit into the FZ on top of stack
173 procedure Ins_Hex_Digit(Digit : in Nibble) is
174 Overflow : WBool := 0;
175 begin
176
177 -- Insert the given nibble, and detect any overflow:
178 FFA_FZ_Insert_Bottom_Nibble(N => Stack(SP),
179 D => Digit,
180 Overflow => Overflow);
181
182 -- Constants which exceed the Width are forbidden:
183 if Overflow = 1 then
184 E("Constant Exceeds Bitness!");
185 end if;
186
187 end;
188
189
190 -- Emit an ASCII representation of N to the terminal
191 procedure Print_FZ(N : in FZ) is
192 S : String(1 .. FFA_FZ_ASCII_Length(N)); -- Mandatorily, exact size
193 begin
194 FFA_FZ_To_Hex_String(N, S); -- Convert N to ASCII hex
195 Write_String(S); -- Print the result to stdout
196 Write_Newline; -- Print newline, for clarity.
197 end Print_FZ;
198
199
200 -- Execute a Normal Op
201 procedure Op_Normal(C : in Character) is
202
203 -- Over/underflow output from certain ops
204 F : Word;
205
206 begin
207
208 case C is
209
210 --------------
211 -- Stickies --
212 --------------
213 -- Enter Commented
214 when '(' =>
215 CommLevel := 1;
216
217 -- Exit Commented (but we aren't in it!)
218 when ')' =>
219 E("Mismatched close-comment parenthesis !");
220
221 -- Enter Quoted
222 when '[' =>
223 QuoteLevel := 1;
224
225 -- Exit Quoted (but we aren't in it!)
226 when ']' =>
227 E("Mismatched close-quote bracket !");
228
229 -- Enter a ~taken~ Conditional branch:
230 when '{' =>
231 Want(1);
232 if FFA_FZ_ZeroP(Stack(SP)) = 1 then
233 CondLevel := 1;
234 end if;
235 Drop;
236
237 -- Exit from a ~non-taken~ Conditional branch:
238 -- ... we push a 0, to suppress the 'else' clause
239 when '}' =>
240 Push;
241 FFA_WBool_To_FZ(0, Stack(SP));
242
243 ----------------
244 -- Immediates --
245 ----------------
246
247 -- These operate on the FZ ~currently~ at top of the stack;
248 -- and this means that the stack may NOT be empty.
249
250 when '0' .. '9' =>
251 Want(1);
252 Ins_Hex_Digit(Character'Pos(C) - Character'Pos('0'));
253
254 when 'A' .. 'F' =>
255 Want(1);
256 Ins_Hex_Digit(10 + Character'Pos(C) - Character'Pos('A'));
257
258 when 'a' .. 'f' =>
259 Want(1);
260 Ins_Hex_Digit(10 + Character'Pos(C) - Character'Pos('a'));
261
262 ------------------
263 -- Stack Motion --
264 ------------------
265
266 -- Push a 0 onto the stack
267 when '.' =>
268 Push;
269 FFA_FZ_Clear(Stack(SP));
270
271 -- Dup
272 when '"' =>
273 Want(1);
274 Push;
275 Stack(SP) := Stack(SP - 1);
276
277 -- Drop
278 when '_' =>
279 Want(1);
280 Drop;
281
282 -- Swap
283 when ''' =>
284 Want(2);
285 FFA_FZ_Swap(Stack(SP), Stack(SP - 1));
286
287 -- Over
288 when '`' =>
289 Want(2);
290 Push;
291 Stack(SP) := Stack(SP - 2);
292
293 ----------------
294 -- Predicates --
295 ----------------
296
297 -- Equality
298 when '=' =>
299 Want(2);
300 FFA_WBool_To_FZ(FFA_FZ_EqP(X => Stack(SP),
301 Y => Stack(SP - 1)),
302 Stack(SP - 1));
303 Drop;
304
305 -- Less-Than
306 when '<' =>
307 Want(2);
308 FFA_WBool_To_FZ(FFA_FZ_LessThanP(X => Stack(SP - 1),
309 Y => Stack(SP)),
310 Stack(SP - 1));
311 Drop;
312
313 -- Greater-Than
314 when '>' =>
315 Want(2);
316 FFA_WBool_To_FZ(FFA_FZ_GreaterThanP(X => Stack(SP - 1),
317 Y => Stack(SP)),
318 Stack(SP - 1));
319 Drop;
320
321 ----------------
322 -- Arithmetic --
323 ----------------
324
325 -- Subtract
326 when '-' =>
327 Want(2);
328 FFA_FZ_Subtract(X => Stack(SP - 1),
329 Y => Stack(SP),
330 Difference => Stack(SP - 1),
331 Underflow => F);
332 Flag := FFA_Word_NZeroP(F);
333 Drop;
334
335 -- Add
336 when '+' =>
337 Want(2);
338 FFA_FZ_Add(X => Stack(SP - 1),
339 Y => Stack(SP),
340 Sum => Stack(SP - 1),
341 Overflow => F);
342 Flag := FFA_Word_NZeroP(F);
343 Drop;
344
345 -- Divide and give Quotient and Remainder
346 when '\' =>
347 Want(2);
348 MustNotZero(Stack(SP));
349 FFA_FZ_IDiv(Dividend => Stack(SP - 1),
350 Divisor => Stack(SP),
351 Quotient => Stack(SP - 1),
352 Remainder => Stack(SP));
353
354 -- Divide and give Quotient only
355 when '/' =>
356 Want(2);
357 MustNotZero(Stack(SP));
358 FFA_FZ_Div(Dividend => Stack(SP - 1),
359 Divisor => Stack(SP),
360 Quotient => Stack(SP - 1));
361 Drop;
362
363 -- Divide and give Remainder only
364 when '%' =>
365 Want(2);
366 MustNotZero(Stack(SP));
367 FFA_FZ_Mod(Dividend => Stack(SP - 1),
368 Divisor => Stack(SP),
369 Remainder => Stack(SP - 1));
370 Drop;
371
372 -- Multiply, give bottom and top halves
373 when '*' =>
374 Want(2);
375 FFA_FZ_Multiply(X => Stack(SP - 1),
376 Y => Stack(SP),
377 XY_Lo => Stack(SP - 1),
378 XY_Hi => Stack(SP));
379
380 -- Modular Multiplication
381 when 'M' =>
382 Want(3);
383 MustNotZero(Stack(SP));
384 FFA_FZ_Modular_Multiply(X => Stack(SP - 2),
385 Y => Stack(SP - 1),
386 Modulus => Stack(SP),
387 Product => Stack(SP - 2));
388 Drop;
389 Drop;
390
391 -- Modular Exponentiation
392 when 'X' =>
393 Want(3);
394 MustNotZero(Stack(SP));
395 FFA_FZ_Modular_Exponentiate(Base => Stack(SP - 2),
396 Exponent => Stack(SP - 1),
397 Modulus => Stack(SP),
398 Result => Stack(SP - 2));
399 Drop;
400 Drop;
401
402 -----------------
403 -- Bitwise Ops --
404 -----------------
405
406 -- Bitwise-And
407 when '&' =>
408 Want(2);
409 FFA_FZ_And(X => Stack(SP - 1),
410 Y => Stack(SP),
411 Result => Stack(SP - 1));
412 Drop;
413
414 -- Bitwise-Or
415 when '|' =>
416 Want(2);
417 FFA_FZ_Or(X => Stack(SP - 1),
418 Y => Stack(SP),
419 Result => Stack(SP - 1));
420 Drop;
421
422 -- Bitwise-Xor
423 when '^' =>
424 Want(2);
425 FFA_FZ_Xor(X => Stack(SP - 1),
426 Y => Stack(SP),
427 Result => Stack(SP - 1));
428 Drop;
429
430 -- Bitwise-Not (1s-Complement)
431 when '~' =>
432 Want(1);
433 FFA_FZ_Not(Stack(SP), Stack(SP));
434
435 -----------
436 -- Other --
437 -----------
438
439 -- Push a FZ of RNGolade onto the stack
440 when '?' =>
441 Push;
442 FFA_FZ_Clear(Stack(SP));
443 FZ_Random(RNG, Stack(SP));
444
445 -- mUx
446 when 'U' =>
447 Want(3);
448 FFA_FZ_Mux(X => Stack(SP - 2),
449 Y => Stack(SP - 1),
450 Result => Stack(SP - 2),
451 Sel => FFA_FZ_NZeroP(Stack(SP)));
452 Drop;
453 Drop;
454
455 -- Put the Overflow flag on the stack
456 when 'O' =>
457 Push;
458 FFA_WBool_To_FZ(Flag, Stack(SP));
459
460 -- Print the FZ on the top of the stack
461 when '#' =>
462 Want(1);
463 Print_FZ(Stack(SP));
464 Drop;
465
466 -- Zap (reset)
467 when 'Z' =>
468 Zap;
469
470 -- Quit with Stack Trace
471 when 'Q' =>
472 for I in reverse Stack'First + 1 .. SP loop
473 Print_FZ(Stack(I));
474 end loop;
475 Quit(0);
476
477 ----------
478 -- NOPs --
479 ----------
480
481 -- Ops we have not yet spoken of -- do nothing
482 when others =>
483 null;
484
485 end case;
486
487 end Op_Normal;
488
489
490 -- Process a Symbol
491 procedure Op(C : in Character) is
492 begin
493 -- First, see whether we are in a state of nestedness:
494
495 -- ... in a Comment block:
496 if CommLevel > 0 then
497 case C is
498 when ')' => -- Drop a nesting level:
499 CommLevel := CommLevel - 1;
500 when '(' => -- Add a nesting level:
501 CommLevel := CommLevel + 1;
502 when others =>
503 null; -- Other symbols have no effect at all
504 end case;
505
506 -- ... in a Quote block:
507 elsif QuoteLevel > 0 then
508 case C is
509 when ']' => -- Drop a nesting level:
510 QuoteLevel := QuoteLevel - 1;
511 when '[' => -- Add a nesting level:
512 QuoteLevel := QuoteLevel + 1;
513 when others =>
514 null; -- Other symbols have no effect on the level
515 end case;
516
517 -- If we aren't the mode-exiting ']', print current symbol:
518 if QuoteLevel > 0 then
519 Write_Char(C);
520 end if;
521
522 --- ... in a ~taken~ Conditional branch:
523 elsif CondLevel > 0 then
524 case C is
525 when '}' => -- Drop a nesting level:
526 CondLevel := CondLevel - 1;
527
528 -- If we exited the Conditional as a result,
529 -- we push a 1 to trigger the possible 'else' clause:
530 if CondLevel = 0 then
531 Push;
532 FFA_WBool_To_FZ(1, Stack(SP));
533 end if;
534
535 when '{' => -- Add a nesting level:
536 CondLevel := CondLevel + 1;
537 when others =>
538 null; -- Other symbols have no effect on the level
539 end case;
540 else
541 -- This is a Normal Op, so proceed with the normal rules.
542 Op_Normal(C);
543 end if;
544
545 end Op;
546
547
548 -- Current Character
549 C : Character;
550
551 begin
552 -- Reset the Calculator
553 Zap;
554 -- Process characters until EOF:
555 loop
556 if Read_Char(C) then
557 -- Execute Op:
558 Op(C);
559 -- Advance Odometer
560 Pos := Pos + 1;
561 else
562 Zap;
563 Quit(0); -- if EOF, we're done
564 end if;
565 end loop;
566 end;
567
568 end FFA_Calc;