Merge branch 'jdw/degree_default'
This commit is contained in:
commit
25db3835e0
261
README.md
261
README.md
@ -1,10 +1,30 @@
|
|||||||
# STC DIY Calculator Firmware
|
# STC DIY Calculator Firmware
|
||||||
|
|
||||||
This is a replacement firmware for the [diyleyuan calculator kit](http://www.diyleyuan.com/jc/L8Q.html). The calculator kit is available for purchase for less than $13 shipped from eBay by searching for "diy calculator kit" (price has increased recently, currently closer to $18 shipped). You will have to solder the kit yourself (see "hardware connections" below). The calculator uses an STC IAP15W413AS microcontroller (an 8051 instruction set-compatible microcontroller) with a built-in serial-port bootloader. See the series [summary](http://www.stcmicro.com/datasheet/STC15W408AS_Features.pdf) and full english [datasheet](https://www.stcmicro.com/datasheet/STC15F2K60S2-en.pdf). This project uses [SDCC](http://sdcc.sourceforge.net/) to compile the C code and [stcgal](https://github.com/grigorig/stcgal) to load the new firmware.
|
This is a replacement firmware for the [diyleyuan calculator kit](http://www.diyleyuan.com/jc/L8Q.html).
|
||||||
|
The calculator kit is available for purchase for less than $13 shipped from eBay by searching for "diy calculator kit"
|
||||||
|
(price has increased recently, currently closer to $18 shipped).
|
||||||
|
You will have to solder the kit yourself (see "hardware connections" below).
|
||||||
|
The calculator uses an STC IAP15W413AS microcontroller
|
||||||
|
(an 8051 instruction set-compatible microcontroller)
|
||||||
|
with a built-in serial-port bootloader.
|
||||||
|
See the series [summary](http://www.stcmicro.com/datasheet/STC15W408AS_Features.pdf)
|
||||||
|
and full english [datasheet](https://www.stcmicro.com/datasheet/STC15F2K60S2-en.pdf).
|
||||||
|
This project uses [SDCC](http://sdcc.sourceforge.net/) to compile the C code
|
||||||
|
and [stcgal](https://github.com/grigorig/stcgal) to load the new firmware.
|
||||||
|
|
||||||
The replacement firmware supports floating-point calculations (using 18 decimal digits plus exponent for arithmetic) with a 4-level RPN stack. Functions include basic arithmetic as well as log(), exp(), y^x, 1/x and sqrt(), all in floating point. (The original firmware supported only fixed-point calculations in chain mode.) I have not added in the resistor value calculator or the decimal/hexadecimal converter features from the original firmware.
|
The replacement firmware supports floating-point calculations
|
||||||
|
(using 18 decimal digits plus exponent for arithmetic)
|
||||||
|
with a 4-level RPN stack.
|
||||||
|
Functions include basic arithmetic as well as log(), exp(), y^x, 1/x and sqrt(),
|
||||||
|
all in floating point.
|
||||||
|
(The original firmware supported only fixed-point calculations in chain mode.)
|
||||||
|
I have not added in the resistor value calculator or the decimal/hexadecimal converter features from the original firmware.
|
||||||
|
|
||||||
Note that once you change the firmware on the calculator, there's no way to go back to the original firmware (the original firmware isn't posted for download anywhere). STC's bootloader on the microcontroller deliberately prevents readback of the microcontroller's content, and STC considers this to be a "feature".
|
Note that once you change the firmware on the calculator,
|
||||||
|
there's no way to go back to the original firmware
|
||||||
|
(the original firmware isn't posted for download anywhere).
|
||||||
|
STC's bootloader on the microcontroller deliberately prevents readback of the microcontroller's content,
|
||||||
|
and STC considers this to be a "feature".
|
||||||
|
|
||||||
Here's a picture of the assembled calculator kit running the new firmware:
|
Here's a picture of the assembled calculator kit running the new firmware:
|
||||||
|
|
||||||
@ -28,9 +48,15 @@ The calculator uses RPN. Calculate (2+3)/(9^2) by pressing the following keys:
|
|||||||
- `*`
|
- `*`
|
||||||
- `÷`
|
- `÷`
|
||||||
|
|
||||||
The = key is used for Enter. There is automatic stack lift so that `9`, `Enter`, `*` is equivalent to 9^2. The stack is a classic 4-level RPN stack, where the T register automatically duplicates.
|
The = key is used for Enter.
|
||||||
|
There is automatic stack lift so that
|
||||||
|
`9`, `Enter`, `*`
|
||||||
|
is equivalent to 9^2.
|
||||||
|
The stack is a classic 4-level RPN stack,
|
||||||
|
where the T register automatically duplicates.
|
||||||
|
|
||||||
The decimal key also doubles as the enter exponent key. For example the following calculates 3E8 / 1550E-9:
|
The decimal key also doubles as the enter exponent key.
|
||||||
|
For example the following calculates 3E8 / 1550E-9:
|
||||||
|
|
||||||
- 3
|
- 3
|
||||||
- .
|
- .
|
||||||
@ -52,7 +78,11 @@ The decimal key also doubles as the enter exponent key. For example the followin
|
|||||||
- 9
|
- 9
|
||||||
- `÷`
|
- `÷`
|
||||||
|
|
||||||
There is currently no way to force the calculator to display in scientific mode. For extremely large numbers that are hard to read, taking the log base 10 of a number will give its exponent. Numbers larger than 18 digits will automatically be displayed in scientific notation, as will numbers smaller than 1E-3.
|
There is currently no way to force the calculator to display in scientific mode.
|
||||||
|
For extremely large numbers that are hard to read,
|
||||||
|
taking the log base 10 of a number will give its exponent.
|
||||||
|
Numbers larger than 18 digits will automatically be displayed in scientific notation,
|
||||||
|
as will numbers smaller than 1E-3.
|
||||||
|
|
||||||
## Keys
|
## Keys
|
||||||
Some of the keys have slightly different functions, see the picture of the emulator Qt GUI.
|
Some of the keys have slightly different functions, see the picture of the emulator Qt GUI.
|
||||||
@ -91,25 +121,45 @@ The keys on the *original* calculator map as follows:
|
|||||||
- acts as acos(x) when shifted down
|
- acts as acos(x) when shifted down
|
||||||
- `3 `: acts as tan(x) when shifted
|
- `3 `: acts as tan(x) when shifted
|
||||||
- acts as atan(x) when shifted down
|
- acts as atan(x) when shifted down
|
||||||
- all trig functions are currently calculated in radians (TODO: change to degrees by default)
|
- all trig functions are calculated in degrees
|
||||||
- `- `: acts as to radians when shifted
|
- `- `: acts as to degrees when shifted
|
||||||
- acts as to degrees when shifted down
|
- acts as to radians when shifted down
|
||||||
- `+ `: acts as LastX when shifted
|
- `+ `: acts as LastX when shifted
|
||||||
- `0 `: acts as off button when shifted
|
- `0 `: acts as off button when shifted
|
||||||
|
|
||||||
|
|
||||||
## Floating Point
|
## Floating Point
|
||||||
|
|
||||||
The calculator internally calculates with an 18 digit significand for better precision, even though at most 16 digits can be displayed. The exponent display is fixed at 2 digits (when it is displayed), but the calculator doesn't prevent you from doing certain operations (e.g. basic arithmetic) which result in numbers with larger exponents.
|
The calculator internally calculates with an 18 digit significand for better precision,
|
||||||
|
even though at most 16 digits can be displayed.
|
||||||
|
The exponent display is fixed at 2 digits (when it is displayed),
|
||||||
|
but the calculator doesn't prevent you from doing certain operations
|
||||||
|
(e.g. basic arithmetic) which result in numbers with larger exponents.
|
||||||
|
|
||||||
Internally, the calculator dedicates 15 bits for representing the signed exponent, so exponents up to +/- 16,383 can be represented (see the internals section below for more information). This is to ensure that intermediate parts of certain calculations (mainly taking the reciprocal of a number) do not prematurely cause overflow or underflow, even when the result is fully representable with just 2 digits. You can do calculations with greater than 2 digits in the exponent, but only 2 digits will be displayed. For larger exponents, a 10 in the ten's place of the exponent will be displayed as a '`:`'. (This just so happens to be the next character after '`9`' in the 1602 LCD's character map).
|
Internally, the calculator dedicates 15 bits for representing the signed exponent,
|
||||||
|
so exponents up to +/- 16,383 can be represented
|
||||||
|
(see the internals section below for more information).
|
||||||
|
This is to ensure that intermediate parts of certain calculations
|
||||||
|
(mainly taking the reciprocal of a number)
|
||||||
|
do not prematurely cause overflow or underflow,
|
||||||
|
even when the result is fully representable with just 2 digits.
|
||||||
|
You can do calculations with greater than 2 digits in the exponent,
|
||||||
|
but only 2 digits will be displayed.
|
||||||
|
For larger exponents,
|
||||||
|
a 10 in the ten's place of the exponent will be displayed as a '`:`'.
|
||||||
|
(This just so happens to be the next character after '`9`' in the 1602 LCD's character map).
|
||||||
|
|
||||||
## Turning off
|
## Turning off
|
||||||
Press `Shift` (the `mode` key on the physical calculator) and then `0` to turn off. On older stc_rpncalc firmwares, or if the calculator is unresponsive, hold `Shift` (the `mode` key on the physical calculator) and `0` *at the same time* to turn off. NOTE: There is no auto power off.
|
Press `Shift` (the `mode` key on the physical calculator) and then `0` to turn off.
|
||||||
|
On older stc_rpncalc firmwares,
|
||||||
|
or if the calculator is unresponsive,
|
||||||
|
hold `Shift` (the `mode` key on the physical calculator) and `0` *at the same time* to turn off.
|
||||||
|
NOTE: There is no auto power off.
|
||||||
|
|
||||||
|
|
||||||
# Building
|
# Building
|
||||||
Github releases has prebuilt binaries for the calculator. Building is fairly straigtforward though.
|
Github releases has prebuilt binaries for the calculator.
|
||||||
|
Building is fairly straigtforward though.
|
||||||
|
|
||||||
- Use the Makefile for building a new firmware for the calculator.
|
- Use the Makefile for building a new firmware for the calculator.
|
||||||
- type `make` to build
|
- type `make` to build
|
||||||
@ -126,7 +176,18 @@ Github releases has prebuilt binaries for the calculator. Building is fairly str
|
|||||||
- `ninja`
|
- `ninja`
|
||||||
|
|
||||||
# Installing
|
# Installing
|
||||||
Note that once you change the firmware on the calculator, it isn't possible to go back to the original firmware. The STC microcontroller used has a bootloader permanently stored in ROM that allows downloading new firmware over a serial port (but not reading the existing firmware). You can re-program it using a USB-to-logic-level-serial (5V) dongle, and the stcgal program. WARNING: a lot of USB-to-logic-level-serial dongles are for 3.3V logic levels instead of the 5V needed. Also note that this is a "logic-level" serial port, and not RS232 levels, which are generally a high negative voltage. The diyleyuan calculator runs at 5V to make it easier to power/drive the LCD display. You have a couple of options:
|
Note that once you change the firmware on the calculator,
|
||||||
|
it isn't possible to go back to the original firmware.
|
||||||
|
The STC microcontroller used has a bootloader permanently stored in ROM that allows downloading new firmware over a serial port
|
||||||
|
(but not reading the existing firmware).
|
||||||
|
You can re-program it using a USB-to-logic-level-serial (5V) dongle,
|
||||||
|
and the stcgal program.
|
||||||
|
WARNING: a lot of USB-to-logic-level-serial dongles are for 3.3V logic levels instead of the 5V needed.
|
||||||
|
Also note that this is a "logic-level" serial port,
|
||||||
|
and not RS232 levels,
|
||||||
|
which are generally a high negative voltage.
|
||||||
|
The diyleyuan calculator runs at 5V to make it easier to power/drive the LCD display.
|
||||||
|
You have a couple of options:
|
||||||
|
|
||||||
1. Buy a USB to logic-level serial dongle that supports 5V operation (these dongles may have a jumper you need to set to switch between 3.3V and 5V). This is the best option.
|
1. Buy a USB to logic-level serial dongle that supports 5V operation (these dongles may have a jumper you need to set to switch between 3.3V and 5V). This is the best option.
|
||||||
- Here is one that works: https://www.amazon.com/gp/product/B00N4MCS1A/
|
- Here is one that works: https://www.amazon.com/gp/product/B00N4MCS1A/
|
||||||
@ -145,11 +206,20 @@ Note that once you change the firmware on the calculator, it isn't possible to g
|
|||||||
Connect to Tx of the USB dongle.
|
Connect to Tx of the USB dongle.
|
||||||
- Pin 16 is Tx from the microcontroller (purple wire in the picture). Connect to Rx of the USB dongle.
|
- Pin 16 is Tx from the microcontroller (purple wire in the picture). Connect to Rx of the USB dongle.
|
||||||
|
|
||||||
I recommend soldering Tx and Rx wires into the plated through holes on the PCB while soldering up the kit, so that the connections are more permanent. I soldered the wires from the back side (leave space for the screw hole). Note that you must "cross over" Tx/Rx going between the microcontroller and the USB dongle (i.e. Rx on the microcontroller goes to Tx on the USB dongle, and Tx on the microcontroller goes to Rx on the USB dongle).
|
I recommend soldering Tx and Rx wires into the plated through holes on the PCB while soldering up the kit,
|
||||||
|
so that the connections are more permanent.
|
||||||
|
I soldered the wires from the back side (leave space for the screw hole).
|
||||||
|
Note that you must "cross over" Tx/Rx going between the microcontroller and the USB dongle
|
||||||
|
(i.e. Rx on the microcontroller goes to Tx on the USB dongle,
|
||||||
|
and Tx on the microcontroller goes to Rx on the USB dongle).
|
||||||
|
|
||||||
![connections_back](./connections_back.jpg)
|
![connections_back](./connections_back.jpg)
|
||||||
|
|
||||||
You must also connect ground to the dongle. A good point to use is header P1. (You may optionally power the calculator with +5V from the USB dongle instead of using button-cell batteries. Header P1 is again a good location to use.) To program the calculator see the "Programming with stcgal" section below.
|
You must also connect ground to the dongle.
|
||||||
|
A good point to use is header P1.
|
||||||
|
(You may optionally power the calculator with +5V from the USB dongle instead of using button-cell batteries.
|
||||||
|
Header P1 is again a good location to use.)
|
||||||
|
To program the calculator see the "Programming with stcgal" section below.
|
||||||
|
|
||||||
Pin on calculator | Color used | Pin on usb-to-serial dongle
|
Pin on calculator | Color used | Pin on usb-to-serial dongle
|
||||||
------------------|------------|-----------------------------
|
------------------|------------|-----------------------------
|
||||||
@ -162,27 +232,55 @@ P1 pin 2 (Gnd) | Black | (Gnd)
|
|||||||
|
|
||||||
#### Voltage regulator
|
#### Voltage regulator
|
||||||
|
|
||||||
Be careful when working on the calculator. The 7550 voltage regulator used has no short circuit protection. It does have a low quiescent current and extremely low dropout voltage though, and you must match the low dropout voltage if replacing the regulator. If you do end up damaging the regulator (like I did), a good replacement is the Microchip MCP1700-5002E/TO. (In the picture though, I just removed the 7550 voltage regulator, shorted pins 2 and 3, and added a capacitor between pin 1 and pins 2/3. I am powering the calculator externally, instead of with batteries.)
|
Be careful when working on the calculator.
|
||||||
|
The 7550 voltage regulator used has no short circuit protection.
|
||||||
|
It does have a low quiescent current and extremely low dropout voltage though,
|
||||||
|
and you must match the low dropout voltage if replacing the regulator.
|
||||||
|
If you do end up damaging the regulator (like I did),
|
||||||
|
a good replacement is the Microchip MCP1700-5002E/TO.
|
||||||
|
(In the picture though, I just removed the 7550 voltage regulator, shorted pins 2 and 3,
|
||||||
|
and added a capacitor between pin 1 and pins 2/3.
|
||||||
|
I am powering the calculator externally, instead of with batteries.)
|
||||||
|
|
||||||
#### Parasitic powering through the usb-to-serial adapter
|
#### Parasitic powering through the usb-to-serial adapter
|
||||||
|
|
||||||
If you have the usb-to-serial adapter connected, the calculator may draw power parasitically through pin 15 (Rx) of the microcontroller. This may prevent the calculator and microcontroller from fully turning off. This may prevent the bootloader from running (which would prevent you from reprogramming the calculator), since the bootloader only runs on power on. To prevent this, use a diode on pin 15 (see image above, or consult the microcontroller datasheet). You may also optionally add a 330 ohm resistor on pin 16 (Tx) of the microcontroller to provide isolation on that pin. (I don't think it's necessary to prevent parasitic power though.)
|
If you have the usb-to-serial adapter connected,
|
||||||
|
the calculator may draw power parasitically through pin 15 (Rx) of the microcontroller.
|
||||||
|
This may prevent the calculator and microcontroller from fully turning off.
|
||||||
|
This may prevent the bootloader from running
|
||||||
|
(which would prevent you from reprogramming the calculator),
|
||||||
|
since the bootloader only runs on power on.
|
||||||
|
To prevent this, use a diode on pin 15
|
||||||
|
(see image above, or consult the microcontroller datasheet).
|
||||||
|
You may also optionally add a 330 ohm resistor on pin 16 (Tx)
|
||||||
|
of the microcontroller to provide isolation on that pin.
|
||||||
|
(I don't think it's necessary to prevent parasitic power though.)
|
||||||
|
|
||||||
#### Adding programming connections after the fact
|
#### Adding programming connections after the fact
|
||||||
|
|
||||||
If you have already soldered the kit together without adding the Tx/Rx serial wires, you can still solder wires directly to the microcontroller fairly easily.
|
If you have already soldered the kit together without adding the Tx/Rx serial wires,
|
||||||
|
you can still solder wires directly to the microcontroller fairly easily.
|
||||||
|
|
||||||
![connections](./connections.jpg)
|
![connections](./connections.jpg)
|
||||||
|
|
||||||
#### Schematic
|
#### Schematic
|
||||||
|
|
||||||
Here is the schematic from the diyleyuan website. Note that the schematic symbol for the microcontroller mistakenly labels P5.4 as P0.0, and mistakenly labels P5.5 as P0.1. The net name labels are correct.
|
Here is the schematic from the diyleyuan website.
|
||||||
|
Note that the schematic symbol for the microcontroller mistakenly labels P5.4 as P0.0,
|
||||||
|
and mistakenly labels P5.5 as P0.1.
|
||||||
|
The net name labels are correct.
|
||||||
|
|
||||||
![schematic](./schematic.gif)
|
![schematic](./schematic.gif)
|
||||||
|
|
||||||
#### Soft-latching power switch
|
#### Soft-latching power switch
|
||||||
|
|
||||||
The soft-latching power switch works as follows: Initially the calculator is off. Both Q1 and Q2 are off. Pressing the On key (S4) turns on Q1 through R1 and D2. Q1 then supplies 5V to the system. Once the microcontroller has power and starts running, it turns on Q2, which keeps Q1 on through R1. To turn off, the microcontroller turns off Q2, which in turn will turn off Q1.
|
The soft-latching power switch works as follows:
|
||||||
|
Initially the calculator is off. Both Q1 and Q2 are off.
|
||||||
|
Pressing the On key (S4) turns on Q1 through R1 and D2.
|
||||||
|
Q1 then supplies 5V to the system.
|
||||||
|
Once the microcontroller has power and starts running,
|
||||||
|
it turns on Q2, which keeps Q1 on through R1.
|
||||||
|
To turn off, the microcontroller turns off Q2, which in turn will turn off Q1.
|
||||||
|
|
||||||
#### Component layout
|
#### Component layout
|
||||||
|
|
||||||
@ -192,21 +290,47 @@ Here is the component layout from the diyleyuan website.
|
|||||||
|
|
||||||
#### Keyswitches, LCD, and other recommended replacement components
|
#### Keyswitches, LCD, and other recommended replacement components
|
||||||
|
|
||||||
The switches used are a knockoff of the Omron B3F series. A good replacement is the B3F-5050 which requires only 130 grams force to depress. The switches included with the kit take more force to depress. (This is somewhat a matter of personal preference though.) I recommend using four 11mm M2 standoffs along with eight M2 screws instead of using the four long screws and nuts provided to hold the top and bottom together. The problem with using the long screws is that in order to tighten them tight enough that they don't become loose easily, the top plate deforms slightly and might interfere with the keys.
|
The switches used are a knockoff of the Omron B3F series.
|
||||||
|
A good replacement is the B3F-5050 which requires only 130 grams force to depress.
|
||||||
|
The switches included with the kit take more force to depress.
|
||||||
|
(This is somewhat a matter of personal preference though.)
|
||||||
|
I recommend using four 11mm M2 standoffs along with eight M2 screws instead of using the four long screws and nuts provided to hold the top and bottom together.
|
||||||
|
The problem with using the long screws is that in order to tighten them tight enough that they don't become loose easily,
|
||||||
|
the top plate deforms slightly and might interfere with the keys.
|
||||||
|
|
||||||
User toml_12953 has made a new keyboard template here (todo: can check into git directly?): https://www.dropbox.com/s/7xkg98ywonp4p1l/New%20DIY%20Template.jpg?dl=1
|
User toml_12953 has made a new keyboard template here (todo: can check into git directly?):
|
||||||
|
https://www.dropbox.com/s/7xkg98ywonp4p1l/New%20DIY%20Template.jpg?dl=1
|
||||||
|
|
||||||
The LCD used is a fairly standard LCD based on a HD44780-compatible controller. The hole spacing for the screw holes on the LCD is 31mm x 75mm. There are many replacements available, including ones that don't need the backlight on to be readable. I recommend a positive transflective or reflective FSTN type, although the one included (transmissive) is very usable with the backlight on. Here is a picture of a positive FSTN display with the backlight off (shown under strong office lighting):
|
The LCD used is a fairly standard LCD based on a HD44780-compatible controller.
|
||||||
|
The hole spacing for the screw holes on the LCD is 31mm x 75mm.
|
||||||
|
There are many replacements available,
|
||||||
|
including ones that don't need the backlight on to be readable.
|
||||||
|
I recommend a positive transflective or reflective FSTN type,
|
||||||
|
although the one included (transmissive) is very usable with the backlight on.
|
||||||
|
Here is a picture of a positive FSTN display with the backlight off
|
||||||
|
(shown under strong office lighting):
|
||||||
|
|
||||||
![backlight off](./no_backlight.jpg)
|
![backlight off](./no_backlight.jpg)
|
||||||
|
|
||||||
The included LCD is a transmissive type, and requires the backlight on to be readable.
|
The included LCD is a transmissive type,
|
||||||
|
and requires the backlight on to be readable.
|
||||||
|
|
||||||
I sometimes use an STC15F2K60S2 for development work. This microcontroller is available in a pin-compatible DIP-28 package for less than $2 a piece in single-digit quantities, and has almost 5 times the flash (a downside is that when programming, the flash takes longer to erase). To program a blank microcontroller, you will have to add the following additional options to `stcgal`: `-l 1200 -t 12000`
|
I sometimes use an STC15F2K60S2 for development work.
|
||||||
|
This microcontroller is available in a pin-compatible DIP-28 package for less than $2 a piece in single-digit quantities,
|
||||||
|
and has almost 5 times the flash
|
||||||
|
(a downside is that when programming, the flash takes longer to erase).
|
||||||
|
To program a blank microcontroller,
|
||||||
|
you will have to add the following additional options to `stcgal`: `-l 1200 -t 12000`
|
||||||
|
|
||||||
## Programming with stcgal
|
## Programming with stcgal
|
||||||
|
|
||||||
Run `stcgal` as shown below, replacing `stc_rpncalc/main.hex` with the actual path to the main.hex you built. There are also prebuilt binaries in the `binaries` directory. In this example, I'm programming at a relatively high line rate of 230,400 bits/s. This works very reliably, but you may want to try at a slower speed to start (omit the `-b 230400` option), especially when using an inline resistor and diode.
|
Run `stcgal` as shown below,
|
||||||
|
replacing `stc_rpncalc/main.hex` with the actual path to the main.hex you built.
|
||||||
|
There are also prebuilt binaries in the Releases section of github.
|
||||||
|
In this example, I'm programming at a relatively high line rate of 230,400 bits/s.
|
||||||
|
This works very reliably, but you may want to try at a slower speed to start
|
||||||
|
(omit the `-b 230400` option),
|
||||||
|
especially when using an inline resistor and diode.
|
||||||
|
|
||||||
~~~~
|
~~~~
|
||||||
$ ./stcgal.py -P stc15 -b 230400 stc_rpncalc/main.hex
|
$ ./stcgal.py -P stc15 -b 230400 stc_rpncalc/main.hex
|
||||||
@ -250,18 +374,43 @@ Disconnected!
|
|||||||
(The name for `stcgal` is probably a play on words from the `avrdude` programming software used to program AVR microcontrollers.)
|
(The name for `stcgal` is probably a play on words from the `avrdude` programming software used to program AVR microcontrollers.)
|
||||||
|
|
||||||
# Bugs
|
# Bugs
|
||||||
1. After division by 0, ln(-), over/underflow, or other operations which give an `Error`, it's possible to still do certain operations on `Error`. Many functions do check, and will not operate on `Error`, but not all of them yet. This is somewhat similar to old soviet Elektronika calculators where `Error` is just a number, and there wasn't enough ROM space to check for errors. (There are people who explore the inner-workings of these calculators by manipulating the `Error` "number".)
|
1. After division by 0, ln(-), over/underflow, or other operations which give an `Error`,
|
||||||
|
it's possible to still do certain operations on `Error`.
|
||||||
|
Many functions do check, and will not operate on `Error`,
|
||||||
|
but not all of them yet.
|
||||||
|
This is somewhat similar to old soviet Elektronika calculators where `Error` is just a number,
|
||||||
|
and there wasn't enough ROM space to check for errors.
|
||||||
|
(There are people who explore the inner-workings of these calculators by manipulating the `Error` "number".)
|
||||||
1. When shifted, keys which do not have a shifted function will instead be interpreted as if there were no shift.
|
1. When shifted, keys which do not have a shifted function will instead be interpreted as if there were no shift.
|
||||||
1. Trigonometric functions are extremely slow and inaccurate.
|
1. Trigonometric functions are extremely slow and inaccurate.
|
||||||
1. There are probably more bugs waiting to be discovered.
|
1. There are probably more bugs waiting to be discovered.
|
||||||
|
|
||||||
# Internals
|
# Internals
|
||||||
## Number Format
|
## Number Format
|
||||||
The original firmware that came with this calculator used a fixed-point format, which significantly limited the range of numbers usable. Additionally, the implementation doesn't actually have enough digits for all displayable results, which can cause errors (e.g. `3,162.277*3,162.28` gives `10,000,010` instead of the correct `10,000,005.31156`).
|
The original firmware that came with this calculator used a fixed-point format,
|
||||||
|
which significantly limited the range of numbers usable.
|
||||||
|
Additionally, the implementation doesn't actually have enough digits for all displayable results,
|
||||||
|
which can cause errors
|
||||||
|
(e.g. `3,162.277*3,162.28` gives `10,000,010` instead of the correct `10,000,005.31156`).
|
||||||
|
|
||||||
This replacement calculator firmware uses decimal floating point, using base-100 to store numbers and do calculations. Base-100 allows for efficient storage into 8-bit bytes, and is easier to work with than packed-BCD. Unlike straight binary representations, base-100 is still fairly easy to display as decimal. Also unlike binary representations, there is no conversion error from binary/decimal (e.g. numbers like `0.1` can be represented exactly).
|
This replacement calculator firmware uses decimal floating point,
|
||||||
|
using base-100 to store numbers and do calculations.
|
||||||
|
Base-100 allows for efficient storage into 8-bit bytes,
|
||||||
|
and is easier to work with than packed-BCD.
|
||||||
|
Unlike straight binary representations,
|
||||||
|
base-100 is still fairly easy to display as decimal.
|
||||||
|
Also unlike binary representations, there is no conversion error from binary/decimal
|
||||||
|
(e.g. numbers like `0.1` can be represented exactly).
|
||||||
|
|
||||||
Each `uint8_t` stores a base-100 "`digit100`", referred to as an "`lsu`", for least significant unit. (The LSU terminology is borrowed from the decNumber library: I originally considered using the decNumber library similar to the WP-34S calculator, but just the library itself takes several times more flash than is available on this calculator. I also considered using the BigNumber arduino library, but that library uses C++ and lots of pointers passed to functions, which are extremely expensive on the 8051 architecture.) The number format is as follows:
|
Each `uint8_t` stores a base-100 "`digit100`",
|
||||||
|
referred to as an "`lsu`", for least significant unit.
|
||||||
|
(The LSU terminology is borrowed from the decNumber library:
|
||||||
|
I originally considered using the decNumber library similar to the WP-34S calculator,
|
||||||
|
but just the library itself takes several times more flash than is available on this calculator.
|
||||||
|
I also considered using the BigNumber arduino library,
|
||||||
|
but that library uses C++ and lots of pointers passed to functions,
|
||||||
|
which are extremely expensive on the 8051 architecture.)
|
||||||
|
The number format is as follows:
|
||||||
|
|
||||||
- `lsu[0]`: contains the most significant `digit100` (the most significant 2 decimal digits)
|
- `lsu[0]`: contains the most significant `digit100` (the most significant 2 decimal digits)
|
||||||
- implicit decimal point between `lsu[0]/10` and `lsu[0]%10`
|
- implicit decimal point between `lsu[0]/10` and `lsu[0]%10`
|
||||||
@ -275,16 +424,23 @@ Each `uint8_t` stores a base-100 "`digit100`", referred to as an "`lsu`", for le
|
|||||||
- range of exponents only needs to be `+/-99`
|
- range of exponents only needs to be `+/-99`
|
||||||
- using 15 bits for the exponent (instead of e.g. 7 bits) prevents certain intermediate results from prematurely causing overflow
|
- using 15 bits for the exponent (instead of e.g. 7 bits) prevents certain intermediate results from prematurely causing overflow
|
||||||
|
|
||||||
For example, the number `13.5` is stored normalized (with no leading zeros in the representation) as follows:
|
For example, the number `13.5` is stored normalized
|
||||||
|
(with no leading zeros in the representation) as follows:
|
||||||
|
|
||||||
- `lsu[0]`: 13
|
- `lsu[0]`: 13
|
||||||
- `lsu[1]`: 50
|
- `lsu[1]`: 50
|
||||||
- `lsu[2]` to `lsu[n-1]`: all 0
|
- `lsu[2]` to `lsu[n-1]`: all 0
|
||||||
- exponent: 1
|
- exponent: 1
|
||||||
|
|
||||||
There is an implicit decimal point between the 1 and 3 in `lsu[0]`, so the number is 1.350 * 10^1, which is equivalent to `13.5`. Similarly, the number `1.35` would be stored the exact same way, except now the exponent is 0.
|
There is an implicit decimal point between the 1 and 3 in `lsu[0]`,
|
||||||
|
so the number is 1.350 * 10^1, which is equivalent to `13.5`.
|
||||||
|
Similarly, the number `1.35` would be stored the exact same way, except now the exponent is 0.
|
||||||
|
|
||||||
The number `0.135` would be stored the same way, except now the exponent is `0x7FFF` (note that the sign bit is 0, and the bottom 15 bits are the 2's complement representation of -1). The number `-13.5` would be stored the same way, except now the exponent is `0x8001` (the sign bit is now 1 which means the number as a whole is negative, but the exponent itself is positive).
|
The number `0.135` would be stored the same way,
|
||||||
|
except now the exponent is `0x7FFF`
|
||||||
|
(note that the sign bit is 0, and the bottom 15 bits are the 2's complement representation of -1).
|
||||||
|
The number `-13.5` would be stored the same way, except now the exponent is `0x8001`
|
||||||
|
(the sign bit is now 1 which means the number as a whole is negative, but the exponent itself is positive).
|
||||||
|
|
||||||
## Arithmetic
|
## Arithmetic
|
||||||
- Addition is done the same way as it's done by hand, although in base-100 instead of decimal.
|
- Addition is done the same way as it's done by hand, although in base-100 instead of decimal.
|
||||||
@ -297,7 +453,7 @@ The number `0.135` would be stored the same way, except now the exponent is `0x7
|
|||||||
## Transcendental Functions
|
## Transcendental Functions
|
||||||
- Logarithms are calculated similar to how it's described by the HP Journal article "Personal Calculator Algorithms IV: Logarithmic Functions" by William Egbert.
|
- Logarithms are calculated similar to how it's described by the HP Journal article "Personal Calculator Algorithms IV: Logarithmic Functions" by William Egbert.
|
||||||
- see `src/decn/proto/ln_mfp.cpp` for initial prototyping development work
|
- see `src/decn/proto/ln_mfp.cpp` for initial prototyping development work
|
||||||
- Exponentials are calculated similar to the HP 35 algorithm, as described [here](http://www.jacques-laporte.org/expx.htm) using the same constants as the logarithm algorithm.
|
- Exponentials are calculated similar to the HP 35 algorithm, as described [here](https://archived.hpcalc.org/laporte/expx.htm) using the same constants as the logarithm algorithm.
|
||||||
- see `src/decn/proto/exp.cpp` for initial prototyping development work
|
- see `src/decn/proto/exp.cpp` for initial prototyping development work
|
||||||
- Powers are calculated using the identity y^x = e^(x*ln(y))
|
- Powers are calculated using the identity y^x = e^(x*ln(y))
|
||||||
- Square roots are calculated using a fixed number of Newton-Raphson iterations to calculatie 1/sqrt(x) and then multiplying by x.
|
- Square roots are calculated using a fixed number of Newton-Raphson iterations to calculatie 1/sqrt(x) and then multiplying by x.
|
||||||
@ -314,24 +470,41 @@ The number `0.135` would be stored the same way, except now the exponent is `0x7
|
|||||||
- Would be nice to have the resistor color band decoder if there is more flash space
|
- Would be nice to have the resistor color band decoder if there is more flash space
|
||||||
- Rounding: currently, to save code space, there is no rounding being done (even for intermediate steps), and numbers are instead truncated. Still, with 18 digits of precision (two guard digits, even if all 16 digits are actually displayed), the results are fairly accurate.
|
- Rounding: currently, to save code space, there is no rounding being done (even for intermediate steps), and numbers are instead truncated. Still, with 18 digits of precision (two guard digits, even if all 16 digits are actually displayed), the results are fairly accurate.
|
||||||
- Square roots could be more-accurately implemented using digit-by-digit methods similar to those described in the HP Journal article "Personal Calculator Algorithms I: Square Roots" by William Egbert.
|
- Square roots could be more-accurately implemented using digit-by-digit methods similar to those described in the HP Journal article "Personal Calculator Algorithms I: Square Roots" by William Egbert.
|
||||||
- calculating using Newton-Raphson iterations for the reciprocal square root 1/sqrt(x), and then multiplying by the original value would probably also be more accurate, and definitely much faster
|
|
||||||
- the iteration for the reciprocal square root is new_estimate = 0.5 * estimate * (3 - x * estimate * estimate)
|
|
||||||
- Reciprocal/division could also be more-accurately implemented using digit-by-digit methods (the Newton-Raphson iterations currently used are quite fast though).
|
- Reciprocal/division could also be more-accurately implemented using digit-by-digit methods (the Newton-Raphson iterations currently used are quite fast though).
|
||||||
|
|
||||||
# Key Debouncing
|
# Key Debouncing
|
||||||
The keyboard matrix is scanned once every 5ms. The keyboard debouncing is based on the quick draw/integrator hybrid algorithm described [here](https://summivox.wordpress.com/2016/06/03/keyboard-matrix-scanning-and-debouncing/). This algorithm combines the advantages of both methods:
|
The keyboard matrix is scanned once every 5ms.
|
||||||
|
The keyboard debouncing is based on the quick draw/integrator hybrid algorithm described
|
||||||
|
[here](https://summivox.wordpress.com/2016/06/03/keyboard-matrix-scanning-and-debouncing/).
|
||||||
|
This algorithm combines the advantages of both methods:
|
||||||
|
|
||||||
1. It signals a key press immediately, the very first instant a keyboard matrix scan detects a key is pressed (similar to the "quick-draw" method).
|
1. It signals a key press immediately, the very first instant a keyboard matrix scan detects a key is pressed (similar to the "quick-draw" method).
|
||||||
1. It has an "integrator" (a saturating up/down counter) to determine both when a key is fully pressed and when a key is fully released. This prevents the mechanically bouncy keys from registering multiple times when pressed.
|
1. It has an "integrator" (a saturating up/down counter) to determine both when a key is fully pressed and when a key is fully released. This prevents the mechanically bouncy keys from registering multiple times when pressed.
|
||||||
|
|
||||||
In practice, the keyboard debouncing works much better than the original firmware (which would occasionally miss keystrokes).
|
In practice, the keyboard debouncing works much better than the original firmware
|
||||||
|
(which would occasionally miss keystrokes).
|
||||||
|
|
||||||
# Implementation on an STC 8051 Microcontroller
|
# Implementation on an STC 8051 Microcontroller
|
||||||
This was my 1st time using an 8051 microcontroller. The architecture is a bit limiting for programming in "high-level" languages such as C compared to more modern architectures -- even compared to other 8-bit architectures such as the AVR (used in the arduino). Most significantly, there is no stack-pointer-relative addressing, which makes C functions takes up a lot of code space, since they have to emulate stack-pointer-relative addressing. Unfortunately, the microcontroller used only has 13K of code space. The compiler used (SDCC) also does not support using a 2nd data pointer, even though STC's implementation of the 8051 has one.
|
This was my 1st time using an 8051 microcontroller.
|
||||||
|
The architecture is a bit limiting for programming in "high-level" languages such as C compared to more modern architectures
|
||||||
|
-- even compared to other 8-bit architectures such as the AVR (used in the arduino).
|
||||||
|
Most significantly, there is no stack-pointer-relative addressing,
|
||||||
|
which makes C functions takes up a lot of code space,
|
||||||
|
since they have to emulate stack-pointer-relative addressing.
|
||||||
|
Unfortunately, the microcontroller used only has 13K of code space.
|
||||||
|
The compiler used (SDCC) also does not support using a 2nd data pointer,
|
||||||
|
even though STC's implementation of the 8051 has one.
|
||||||
|
|
||||||
I've avoided relying on the functions being able to be re-entrant, so that they do not depend on having a stack. SDCC is *not* set to use `--stack-auto` to reduce code size (this means functions are not re-entrant). Some "large" local variables are declared as static in functions to save on the code space needed to emulate a stack. I used a lot more globals than what I would typically like to have used, and a lot less pointers passed to functions, since these are extremely expensive (to account for the 3 different memory types).
|
I've avoided relying on the functions being able to be re-entrant,
|
||||||
|
so that they do not depend on having a stack.
|
||||||
|
SDCC is *not* set to use `--stack-auto` to reduce code size (this means functions are not re-entrant).
|
||||||
|
Some "large" local variables are declared as static in functions to save on the code space needed to emulate a stack.
|
||||||
|
I used a lot more globals than what I would typically like to have used,
|
||||||
|
and a lot less pointers passed to functions,
|
||||||
|
since these are extremely expensive (to account for the 3 different memory types).
|
||||||
|
|
||||||
Another weird thing about the 8051 is that not all of the memory is addressed the same way. On this microcontroller, there are 512 bytes of ram total, of which:
|
Another weird thing about the 8051 is that not all of the memory is addressed the same way.
|
||||||
|
On this microcontroller, there are 512 bytes of ram total, of which:
|
||||||
|
|
||||||
- only 128 bytes can be addressed directly (or indirectly)
|
- only 128 bytes can be addressed directly (or indirectly)
|
||||||
- the start of this address space is also shared with general purpose registers, so you don't actually have the full 128 bytes of directly addressable memory
|
- the start of this address space is also shared with general purpose registers, so you don't actually have the full 128 bytes of directly addressable memory
|
||||||
@ -341,7 +514,9 @@ Another weird thing about the 8051 is that not all of the memory is addressed th
|
|||||||
- on the original 8051, this memory would have actually been external, but on this microcontroller the "external" ram is built in
|
- on the original 8051, this memory would have actually been external, but on this microcontroller the "external" ram is built in
|
||||||
- addressing this memory is more difficult than addressing indirect memory, which is itself more difficult than addressing memory directly
|
- addressing this memory is more difficult than addressing indirect memory, which is itself more difficult than addressing memory directly
|
||||||
|
|
||||||
Thus, there are special compiler directives to tell it what address space to place variables in memory. (Even for a simple calculator, there isn't enough directly addressable memory (128 bytes) to store everything.) General-purpose pointers and operations using general-purpose pointers are relatively expensive since the pointers must encode the memory type.
|
Thus, there are special compiler directives to tell it what address space to place variables in memory.
|
||||||
|
(Even for a simple calculator, there isn't enough directly addressable memory (128 bytes) to store everything.)
|
||||||
|
General-purpose pointers and operations using general-purpose pointers are relatively expensive since the pointers must encode the memory type.
|
||||||
|
|
||||||
|
|
||||||
# Licensing
|
# Licensing
|
||||||
|
BIN
qt_gui.png
BIN
qt_gui.png
Binary file not shown.
Before Width: | Height: | Size: 67 KiB After Width: | Height: | Size: 71 KiB |
@ -168,7 +168,7 @@ ApplicationWindow
|
|||||||
["Shift", "1/x", " √<span style=\"text-decoration: overline\">x</span> ", "CL<i>x</i>"],
|
["Shift", "1/x", " √<span style=\"text-decoration: overline\">x</span> ", "CL<i>x</i>"],
|
||||||
["y<sup>x</sup> ", "ln(x)", "log(x)", "π"],
|
["y<sup>x</sup> ", "ln(x)", "log(x)", "π"],
|
||||||
["R▼", "e<sup>x</sup>", "10<sup>x</sup>", ""],
|
["R▼", "e<sup>x</sup>", "10<sup>x</sup>", ""],
|
||||||
["sin(x)", "cos(x)", "tan(x)", "►rad"],
|
["sin(x)", "cos(x)", "tan(x)", "►deg"],
|
||||||
["off", "STO", "RCL", "LAST<i>x</i>"]
|
["off", "STO", "RCL", "LAST<i>x</i>"]
|
||||||
]
|
]
|
||||||
|
|
||||||
@ -180,7 +180,7 @@ ApplicationWindow
|
|||||||
["", "", "", ""],
|
["", "", "", ""],
|
||||||
["", "", "", ""],
|
["", "", "", ""],
|
||||||
["R▲", "", "", ""],
|
["R▲", "", "", ""],
|
||||||
["asin(x)", "acos(x)", "atan(x)", "►deg"],
|
["asin(x)", "acos(x)", "atan(x)", "►rad"],
|
||||||
["", "", "", ""]
|
["", "", "", ""]
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -116,9 +116,9 @@ void process_cmd(char cmd){
|
|||||||
//////////
|
//////////
|
||||||
case '-':{
|
case '-':{
|
||||||
if (IsShiftedUp) {
|
if (IsShiftedUp) {
|
||||||
do_unary_op(to_radian_decn);
|
|
||||||
} else if (IsShiftedDown) {
|
|
||||||
do_unary_op(to_degree_decn);
|
do_unary_op(to_degree_decn);
|
||||||
|
} else if (IsShiftedDown) {
|
||||||
|
do_unary_op(to_radian_decn);
|
||||||
} else {
|
} else {
|
||||||
negate_decn(&stack(STACK_X));
|
negate_decn(&stack(STACK_X));
|
||||||
do_binary_op(add_decn);
|
do_binary_op(add_decn);
|
||||||
|
@ -85,9 +85,9 @@ const dec80 DECN_LN_10 = {
|
|||||||
0, {23, 2, 58, 50, 92, 99, 40, 45, 68}
|
0, {23, 2, 58, 50, 92, 99, 40, 45, 68}
|
||||||
};
|
};
|
||||||
|
|
||||||
// 2 pi
|
// pi
|
||||||
const dec80 DECN_2PI = {
|
const dec80 DECN_PI = {
|
||||||
0, {62, 83, 18, 53, 7, 17, 95, 86, 48}
|
0, {31, 41, 59, 26, 53, 58, 97, 93, 24}
|
||||||
};
|
};
|
||||||
|
|
||||||
// pi/2
|
// pi/2
|
||||||
@ -1426,8 +1426,8 @@ void sqrt_decn(void){
|
|||||||
#endif //USE_POW_SQRT_IMPL
|
#endif //USE_POW_SQRT_IMPL
|
||||||
|
|
||||||
|
|
||||||
// see W.E. Egbert, "Personal Calculator Algorithms II: Trigonometric functions"
|
// normal angle to between 0 and 360 degrees
|
||||||
void project_decn_into_0_2pi(void) {
|
void normalize_0_360(void) {
|
||||||
const uint8_t is_negative = (AccDecn.exponent < 0);
|
const uint8_t is_negative = (AccDecn.exponent < 0);
|
||||||
exp_t exponent;
|
exp_t exponent;
|
||||||
|
|
||||||
@ -1436,11 +1436,14 @@ void project_decn_into_0_2pi(void) {
|
|||||||
negate_decn(&AccDecn);
|
negate_decn(&AccDecn);
|
||||||
}
|
}
|
||||||
exponent = get_exponent(&AccDecn);
|
exponent = get_exponent(&AccDecn);
|
||||||
copy_decn(&BDecn, &DECN_2PI);
|
//B = 360
|
||||||
|
set_dec80_zero(&BDecn);
|
||||||
|
BDecn.lsu[0] = 36;
|
||||||
|
BDecn.exponent = 2;
|
||||||
if (compare_magn() > 0) {
|
if (compare_magn() > 0) {
|
||||||
do {
|
do {
|
||||||
do {
|
do {
|
||||||
copy_decn(&BDecn, &DECN_2PI);
|
//B = 3.6e...
|
||||||
BDecn.exponent = exponent;
|
BDecn.exponent = exponent;
|
||||||
if (compare_magn() >= 0) {
|
if (compare_magn() >= 0) {
|
||||||
negate_decn(&BDecn);
|
negate_decn(&BDecn);
|
||||||
@ -1450,12 +1453,13 @@ void project_decn_into_0_2pi(void) {
|
|||||||
}
|
}
|
||||||
} while (1);
|
} while (1);
|
||||||
exponent--;
|
exponent--;
|
||||||
} while (exponent >= 0);
|
} while (exponent >= 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_negative) {
|
if (is_negative) {
|
||||||
negate_decn(&AccDecn);
|
negate_decn(&AccDecn);
|
||||||
copy_decn(&BDecn, &DECN_2PI);
|
//B = 360
|
||||||
|
BDecn.exponent = 2;
|
||||||
add_decn();
|
add_decn();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1467,13 +1471,14 @@ void project_decn_into_0_2pi(void) {
|
|||||||
#define THETA Tmp4Decn
|
#define THETA Tmp4Decn
|
||||||
void sincos_decn(const uint8_t sincos_arctan) {
|
void sincos_decn(const uint8_t sincos_arctan) {
|
||||||
const uint8_t is_negative = AccDecn.exponent < 0;
|
const uint8_t is_negative = AccDecn.exponent < 0;
|
||||||
if (sincos_arctan) {
|
if (sincos_arctan) { //calculate arctan
|
||||||
set_dec80_zero(&THETA);
|
set_dec80_zero(&THETA);
|
||||||
if (is_negative) negate_decn(&AccDecn);
|
if (is_negative) negate_decn(&AccDecn);
|
||||||
copy_decn(&COS, &AccDecn);
|
copy_decn(&COS, &AccDecn);
|
||||||
set_decn_one(&SIN);
|
set_decn_one(&SIN);
|
||||||
} else {
|
} else { //calculate sin/cos
|
||||||
project_decn_into_0_2pi();
|
normalize_0_360();
|
||||||
|
to_radian_decn();
|
||||||
copy_decn(&THETA, &AccDecn);
|
copy_decn(&THETA, &AccDecn);
|
||||||
set_decn_one(&COS);
|
set_decn_one(&COS);
|
||||||
set_dec80_zero(&SIN);
|
set_dec80_zero(&SIN);
|
||||||
@ -1482,13 +1487,13 @@ void sincos_decn(const uint8_t sincos_arctan) {
|
|||||||
negate_decn(&SIN);
|
negate_decn(&SIN);
|
||||||
}
|
}
|
||||||
do {
|
do {
|
||||||
if (sincos_arctan) {
|
if (sincos_arctan) { //calculate arctan
|
||||||
// THETA is in AccDecn from previous iteration
|
// THETA is in AccDecn from previous iteration
|
||||||
if (COS.exponent < 0) {
|
if (COS.exponent < 0) {
|
||||||
if (is_negative) negate_decn(&AccDecn);
|
if (is_negative) negate_decn(&AccDecn);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
} else {
|
} else { //calculate sin/cos
|
||||||
if (THETA.exponent < 0) {
|
if (THETA.exponent < 0) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -1539,10 +1544,11 @@ void tan_decn(void) {
|
|||||||
|
|
||||||
void arctan_decn(void) {
|
void arctan_decn(void) {
|
||||||
sincos_decn(1);
|
sincos_decn(1);
|
||||||
|
to_degree_decn();
|
||||||
}
|
}
|
||||||
|
|
||||||
// see W.E. Egbert, "Personal Calculator Algorithms III: Inverse Trigonometric Functions"
|
// see W.E. Egbert, "Personal Calculator Algorithms III: Inverse Trigonometric Functions"
|
||||||
void arcsin_decn(void) {
|
void arcsin_decn_rad(void) {
|
||||||
st_push_decn(&AccDecn);
|
st_push_decn(&AccDecn);
|
||||||
copy_decn(&BDecn, &AccDecn);
|
copy_decn(&BDecn, &AccDecn);
|
||||||
mult_decn();
|
mult_decn();
|
||||||
@ -1553,15 +1559,20 @@ void arcsin_decn(void) {
|
|||||||
recip_decn();
|
recip_decn();
|
||||||
st_pop_decn(&BDecn);
|
st_pop_decn(&BDecn);
|
||||||
mult_decn();
|
mult_decn();
|
||||||
|
|
||||||
sincos_decn(1);
|
sincos_decn(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void arcsin_decn(void) {
|
||||||
|
arcsin_decn_rad();
|
||||||
|
to_degree_decn();
|
||||||
|
}
|
||||||
|
|
||||||
void arccos_decn(void) {
|
void arccos_decn(void) {
|
||||||
arcsin_decn();
|
arcsin_decn_rad();
|
||||||
negate_decn(&AccDecn);
|
negate_decn(&AccDecn);
|
||||||
copy_decn(&BDecn, &DECN_PI2);
|
copy_decn(&BDecn, &DECN_PI2);
|
||||||
add_decn();
|
add_decn();
|
||||||
|
to_degree_decn();
|
||||||
}
|
}
|
||||||
#undef SIN
|
#undef SIN
|
||||||
#undef COS
|
#undef COS
|
||||||
@ -1579,8 +1590,7 @@ void to_radian_decn(void) {
|
|||||||
|
|
||||||
void pi_decn(void) {
|
void pi_decn(void) {
|
||||||
set_dec80_zero(&BDecn);
|
set_dec80_zero(&BDecn);
|
||||||
BDecn.lsu[0] = 5; // 0.5 00 ..
|
copy_decn(&AccDecn, &DECN_PI);
|
||||||
copy_decn(&AccDecn, &DECN_2PI);
|
|
||||||
mult_decn();
|
mult_decn();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -22,6 +22,8 @@
|
|||||||
namespace bmp = boost::multiprecision;
|
namespace bmp = boost::multiprecision;
|
||||||
using Catch::Matchers::Equals;
|
using Catch::Matchers::Equals;
|
||||||
|
|
||||||
|
static const bmp::mpfr_float mPI = boost::math::constants::pi<bmp::mpfr_float>();
|
||||||
|
|
||||||
|
|
||||||
static void trig_test(void (*operation)(void), bmp::mpfr_float (*mpfr_operation)(bmp::mpfr_float x),
|
static void trig_test(void (*operation)(void), bmp::mpfr_float (*mpfr_operation)(bmp::mpfr_float x),
|
||||||
double rtol, double atol)
|
double rtol, double atol)
|
||||||
@ -50,10 +52,10 @@ static void trig_test(void (*operation)(void), bmp::mpfr_float (*mpfr_operation)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void sin_test(double rtol=5e-3, double atol=1e-3)
|
static void sin_test(double rtol=6e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
CAPTURE("sin test");
|
CAPTURE("sin test");
|
||||||
trig_test(sin_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return sin(x);}, rtol, atol);
|
trig_test(sin_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return sin(x * mPI / 180);}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void sin_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void sin_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -64,10 +66,10 @@ static void sin_test(const char* a_str, int a_exp, double rtol=5e-3, double atol
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void cos_test(double rtol=5e-3, double atol=1e-3)
|
static void cos_test(double rtol=6e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
CAPTURE("cos test");
|
CAPTURE("cos test");
|
||||||
trig_test(cos_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return cos(x);}, rtol, atol);
|
trig_test(cos_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return cos(x * mPI / 180);}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void cos_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void cos_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -80,7 +82,7 @@ static void cos_test(const char* a_str, int a_exp, double rtol=5e-3, double atol
|
|||||||
|
|
||||||
static void tan_test(double rtol=5e-3, double atol=1e-3)
|
static void tan_test(double rtol=5e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
trig_test(tan_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return tan(x);}, rtol, atol);
|
trig_test(tan_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return tan(x * mPI / 180);}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tan_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void tan_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -93,7 +95,7 @@ static void tan_test(const char* a_str, int a_exp, double rtol=5e-3, double atol
|
|||||||
|
|
||||||
static void atan_test(double rtol=5e-3, double atol=1e-3)
|
static void atan_test(double rtol=5e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
trig_test(arctan_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return atan(x);}, rtol, atol);
|
trig_test(arctan_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return atan(x)*180/mPI;}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void atan_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void atan_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -106,7 +108,7 @@ static void atan_test(const char* a_str, int a_exp, double rtol=5e-3, double ato
|
|||||||
|
|
||||||
static void asin_test(double rtol=5e-3, double atol=1e-3)
|
static void asin_test(double rtol=5e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
trig_test(arcsin_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return asin(x);}, rtol, atol);
|
trig_test(arcsin_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return asin(x)*180/mPI;}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void asin_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void asin_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -119,7 +121,7 @@ static void asin_test(const char* a_str, int a_exp, double rtol=5e-3, double ato
|
|||||||
|
|
||||||
static void acos_test(double rtol=5e-3, double atol=1e-3)
|
static void acos_test(double rtol=5e-3, double atol=1e-3)
|
||||||
{
|
{
|
||||||
trig_test(arccos_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return acos(x);}, rtol, atol);
|
trig_test(arccos_decn, [](bmp::mpfr_float x) -> bmp::mpfr_float {return acos(x)*180/mPI;}, rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void acos_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
static void acos_test(const char* a_str, int a_exp, double rtol=5e-3, double atol=1e-3)
|
||||||
@ -129,101 +131,175 @@ static void acos_test(const char* a_str, int a_exp, double rtol=5e-3, double ato
|
|||||||
acos_test(rtol, atol);
|
acos_test(rtol, atol);
|
||||||
}
|
}
|
||||||
|
|
||||||
const char * const pi = "3.141592653589793239";
|
// const char * const pi = "3.141592653589793239";
|
||||||
const char * const pi_threequarters = "2.356194490192344929";
|
// const char * const pi_threequarters = "2.356194490192344929";
|
||||||
const char * const pi_halved = "1.570796326794896619";
|
// const char * const pi_halved = "1.570796326794896619";
|
||||||
const char * const pi_quarter = ".7853981633974483096";
|
// const char * const pi_quarter = ".7853981633974483096";
|
||||||
|
|
||||||
|
|
||||||
TEST_CASE("sin") {
|
TEST_CASE("sin") {
|
||||||
sin_test("0.1", 0);
|
sin_test("0.1", 0, 0.2);
|
||||||
sin_test("0.05", 0, 1e-2);
|
|
||||||
sin_test("0.01", 0, -1);
|
|
||||||
sin_test("0.001", 0, -1);
|
|
||||||
sin_test("0.0001", 0, -1);
|
|
||||||
sin_test("0.00001", 0, -1);
|
|
||||||
sin_test("0.000001", 0, -1);
|
|
||||||
sin_test("0.0", 0, -1);
|
sin_test("0.0", 0, -1);
|
||||||
sin_test("0.2", 0);
|
sin_test("1.5", 0, 0.02);
|
||||||
sin_test("0.3", 0);
|
sin_test("2.0", 0, 0.02);
|
||||||
sin_test("0.4", 0);
|
sin_test("2.5", 0, 0.02);
|
||||||
sin_test("0.9", 0);
|
sin_test("3.0", 0, 0.02);
|
||||||
sin_test("1.5", 0);
|
sin_test("10", 0);
|
||||||
sin_test("2.0", 0);
|
sin_test("20", 0);
|
||||||
sin_test("2.5", 0);
|
sin_test("30", 0);
|
||||||
sin_test("3.0", 0);
|
sin_test("40", 0);
|
||||||
sin_test(pi, 0, -1);
|
sin_test("80", 0);
|
||||||
sin_test(pi_quarter, 0);
|
sin_test("120", 0);
|
||||||
sin_test(pi_halved, 0);
|
sin_test("160", 0);
|
||||||
sin_test(pi_threequarters, 0);
|
sin_test("200", 0);
|
||||||
|
sin_test("240", 0);
|
||||||
|
sin_test("280", 0);
|
||||||
|
sin_test("320", 0);
|
||||||
|
sin_test("359", 0, 0.02);
|
||||||
|
sin_test("360", 0, -1, 0.001);
|
||||||
|
sin_test("361", 0, 0.02);
|
||||||
|
sin_test("400", 0);
|
||||||
|
// sin_test(pi, 0, -1);
|
||||||
|
// sin_test(pi_quarter, 0);
|
||||||
|
// sin_test(pi_halved, 0);
|
||||||
|
// sin_test(pi_threequarters, 0);
|
||||||
|
sin_test("180.0", 0, -1);
|
||||||
|
sin_test("45.0", 0);
|
||||||
|
sin_test("90.0", 0);
|
||||||
|
sin_test("135.0", 0);
|
||||||
sin_test("1000.0", 0);
|
sin_test("1000.0", 0);
|
||||||
sin_test("-0.5", 0);
|
sin_test("-0.5", 0, 0.2);
|
||||||
sin_test("-1.5", 0);
|
sin_test("-1.5", 0, 0.02);
|
||||||
sin_test("-2.0", 0);
|
sin_test("-2.0", 0, 0.02);
|
||||||
sin_test("-2.5", 0);
|
sin_test("-2.5", 0, 0.02);
|
||||||
sin_test("-3.0", 0);
|
sin_test("-3.0", 0, 0.02);
|
||||||
sin_test("-9.0", 0);
|
sin_test("-9.0", 0);
|
||||||
sin_test("-18.0", 0);
|
sin_test("-18.0", 0);
|
||||||
sin_test("-27.0", 0);
|
sin_test("-27.0", 0);
|
||||||
sin_test("-1000.0", 0);
|
sin_test("-1000.0", 0);
|
||||||
|
sin_test("-30", 0);
|
||||||
|
sin_test("-40", 0);
|
||||||
|
sin_test("-80", 0);
|
||||||
|
sin_test("-120", 0);
|
||||||
|
sin_test("-160", 0);
|
||||||
|
sin_test("-200", 0);
|
||||||
|
sin_test("-240", 0);
|
||||||
|
sin_test("-280", 0);
|
||||||
|
sin_test("-320", 0);
|
||||||
|
sin_test("-360", 0, -1, 0.001);
|
||||||
|
sin_test("-400", 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("cos") {
|
TEST_CASE("cos") {
|
||||||
cos_test("0.1", 0);
|
cos_test("0.1", 0, 0.2);
|
||||||
cos_test("0.05", 0);
|
cos_test("0.0", 0, -1);
|
||||||
cos_test("0.01", 0);
|
cos_test("1.5", 0, 0.02);
|
||||||
cos_test("0.001", 0);
|
cos_test("2.0", 0, 0.02);
|
||||||
cos_test("0.0001", 0);
|
cos_test("2.5", 0, 0.02);
|
||||||
cos_test("0.00001", 0);
|
cos_test("3.0", 0, 0.02);
|
||||||
cos_test("0.000001", 0);
|
cos_test("10", 0);
|
||||||
cos_test("0.0", 0);
|
cos_test("20", 0);
|
||||||
cos_test("0.2", 0);
|
cos_test("30", 0);
|
||||||
cos_test("0.3", 0);
|
cos_test("40", 0);
|
||||||
cos_test("0.4", 0);
|
cos_test("80", 0);
|
||||||
cos_test("0.9", 0);
|
cos_test("120", 0);
|
||||||
cos_test("1.5", 0);
|
cos_test("160", 0);
|
||||||
cos_test("2.0", 0);
|
cos_test("200", 0);
|
||||||
cos_test("2.5", 0);
|
cos_test("240", 0);
|
||||||
cos_test("3.0", 0);
|
cos_test("280", 0, 0.006);
|
||||||
cos_test(pi, 0);
|
cos_test("320", 0);
|
||||||
cos_test(pi_quarter, 0);
|
cos_test("359", 0, 0.02);
|
||||||
cos_test(pi_halved, 0, -1);
|
cos_test("360", 0, -1, 0.001);
|
||||||
cos_test(pi_threequarters, 0);
|
cos_test("361", 0, 0.02);
|
||||||
cos_test("1000.0", 0);
|
cos_test("400", 0);
|
||||||
cos_test("-0.5", 0);
|
// cos_test(pi, 0, -1);
|
||||||
cos_test("-1.5", 0);
|
// cos_test(pi_quarter, 0);
|
||||||
cos_test("-2.0", 0);
|
// cos_test(pi_halved, 0);
|
||||||
cos_test("-2.5", 0);
|
// cos_test(pi_threequarters, 0);
|
||||||
cos_test("-3.0", 0);
|
cos_test("180.0", 0, -1);
|
||||||
|
cos_test("45.0", 0);
|
||||||
|
cos_test("90.0", 0, -1, 0.001);
|
||||||
|
cos_test("135.0", 0);
|
||||||
|
cos_test("1000.0", 0, 0.006);
|
||||||
|
cos_test("-0.5", 0, 0.2);
|
||||||
|
cos_test("-1.5", 0, 0.02);
|
||||||
|
cos_test("-2.0", 0, 0.02);
|
||||||
|
cos_test("-2.5", 0, 0.02);
|
||||||
|
cos_test("-3.0", 0, 0.02);
|
||||||
cos_test("-9.0", 0);
|
cos_test("-9.0", 0);
|
||||||
cos_test("-18.0", 0);
|
cos_test("-18.0", 0);
|
||||||
cos_test("-27.0", 0);
|
cos_test("-27.0", 0);
|
||||||
cos_test("-1000.0", 0);
|
cos_test("-1000.0", 0);
|
||||||
|
cos_test("-30", 0);
|
||||||
|
cos_test("-40", 0);
|
||||||
|
cos_test("-80", 0, 0.006);
|
||||||
|
cos_test("-120", 0);
|
||||||
|
cos_test("-160", 0);
|
||||||
|
cos_test("-200", 0);
|
||||||
|
cos_test("-240", 0);
|
||||||
|
cos_test("-280", 0);
|
||||||
|
cos_test("-320", 0);
|
||||||
|
cos_test("-360", 0, -1, 0.001);
|
||||||
|
cos_test("-400", 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
TEST_CASE("tan") {
|
TEST_CASE("tan") {
|
||||||
tan_test("0.1", 0);
|
tan_test("0.1", 0, 0.2);
|
||||||
tan_test("0.05", 0, 1e-2);
|
|
||||||
tan_test("0.01", 0, 5e-2);
|
|
||||||
tan_test("0.001", 0, -1);
|
|
||||||
tan_test("0.0001", 0, -1);
|
|
||||||
tan_test("0.00001", 0, -1);
|
|
||||||
tan_test("0.000001", 0, -1);
|
|
||||||
tan_test("0.0", 0, -1);
|
tan_test("0.0", 0, -1);
|
||||||
tan_test("0.2", 0);
|
tan_test("1.5", 0, 0.02);
|
||||||
tan_test("0.3", 0);
|
tan_test("2.0", 0, 0.02);
|
||||||
tan_test("0.4", 0);
|
tan_test("2.5", 0, 0.02);
|
||||||
tan_test("0.9", 0);
|
tan_test("3.0", 0, 0.02);
|
||||||
tan_test("1.5", 0);
|
tan_test("10", 0);
|
||||||
tan_test("2.0", 0);
|
tan_test("20", 0);
|
||||||
tan_test("2.5", 0);
|
tan_test("30", 0);
|
||||||
tan_test("3.0", 0);
|
tan_test("40", 0);
|
||||||
|
tan_test("80", 0);
|
||||||
|
tan_test("120", 0);
|
||||||
|
tan_test("160", 0);
|
||||||
|
tan_test("200", 0);
|
||||||
|
tan_test("240", 0);
|
||||||
|
tan_test("280", 0, 0.006);
|
||||||
|
tan_test("320", 0);
|
||||||
|
tan_test("359", 0, 0.02);
|
||||||
|
tan_test("360", 0, -1, 0.001);
|
||||||
|
tan_test("361", 0, 0.02);
|
||||||
|
tan_test("400", 0);
|
||||||
|
// tan_test(pi, 0, -1);
|
||||||
|
// tan_test(pi_quarter, 0);
|
||||||
|
// tan_test(pi_halved, 0);
|
||||||
|
// tan_test(pi_threequarters, 0);
|
||||||
|
tan_test("180.0", 0, -1);
|
||||||
|
tan_test("45.0", 0);
|
||||||
|
tan_test("90.0", 0, 2);
|
||||||
|
tan_test("135.0", 0);
|
||||||
|
tan_test("1000.0", 0, 0.006);
|
||||||
|
tan_test("-0.5", 0, 0.2);
|
||||||
|
tan_test("-1.5", 0, 0.02);
|
||||||
|
tan_test("-2.0", 0, 0.02);
|
||||||
|
tan_test("-2.5", 0, 0.02);
|
||||||
|
tan_test("-3.0", 0, 0.02);
|
||||||
|
tan_test("-9.0", 0);
|
||||||
|
tan_test("-18.0", 0);
|
||||||
|
tan_test("-27.0", 0);
|
||||||
|
tan_test("-1000.0", 0);
|
||||||
|
tan_test("-30", 0);
|
||||||
|
tan_test("-40", 0);
|
||||||
|
tan_test("-80", 0, 0.006);
|
||||||
|
tan_test("-120", 0);
|
||||||
|
tan_test("-160", 0);
|
||||||
|
tan_test("-200", 0);
|
||||||
|
tan_test("-240", 0);
|
||||||
|
tan_test("-280", 0);
|
||||||
|
tan_test("-320", 0);
|
||||||
|
tan_test("-360", 0, -1, 0.001);
|
||||||
|
tan_test("-400", 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("arctan") {
|
TEST_CASE("arctan") {
|
||||||
atan_test("0.001", 0, -1, 2e-3);
|
atan_test("0.001", 0, -1, 0.06);
|
||||||
atan_test("-0.001", 0, -1, 2e-3);
|
atan_test("-0.001", 0, -1, 0.06);
|
||||||
atan_test("0.7", 0);
|
atan_test("0.7", 0);
|
||||||
atan_test("-0.7", 0);
|
atan_test("-0.7", 0);
|
||||||
atan_test("0.1", 0);
|
atan_test("0.1", 0);
|
||||||
@ -234,12 +310,12 @@ TEST_CASE("arctan") {
|
|||||||
atan_test("-2.0", 0);
|
atan_test("-2.0", 0);
|
||||||
atan_test("3.0", 0);
|
atan_test("3.0", 0);
|
||||||
atan_test("-3.0", 0);
|
atan_test("-3.0", 0);
|
||||||
atan_test("0", 0, -1);
|
atan_test("0", 0, -1, 0.06);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE("arcsin") {
|
TEST_CASE("arcsin") {
|
||||||
asin_test("0.001", 0, -1);
|
asin_test("0.001", 0, -1, 0.06);
|
||||||
asin_test("-0.001", 0, -1);
|
asin_test("-0.001", 0, -1, 0.06);
|
||||||
asin_test("0.7", 0);
|
asin_test("0.7", 0);
|
||||||
asin_test("-0.7", 0);
|
asin_test("-0.7", 0);
|
||||||
asin_test("0.1", 0, 1e-2);
|
asin_test("0.1", 0, 1e-2);
|
||||||
@ -265,7 +341,7 @@ static const int NUM_RAND_TRIG_TESTS = 4321; //trig tests are slow
|
|||||||
TEST_CASE("sin random"){
|
TEST_CASE("sin random"){
|
||||||
std::default_random_engine gen;
|
std::default_random_engine gen;
|
||||||
std::uniform_int_distribution<int> distrib(0,99);
|
std::uniform_int_distribution<int> distrib(0,99);
|
||||||
std::uniform_int_distribution<int> exp_distrib(-1,0); //restrict range for now
|
std::uniform_int_distribution<int> exp_distrib(0, 2); //restrict range for now
|
||||||
std::uniform_int_distribution<int> sign_distrib(0,1);
|
std::uniform_int_distribution<int> sign_distrib(0,1);
|
||||||
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
||||||
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
||||||
@ -274,25 +350,27 @@ TEST_CASE("sin random"){
|
|||||||
int exp = exp_distrib(gen);
|
int exp = exp_distrib(gen);
|
||||||
int sign = sign_distrib(gen);
|
int sign = sign_distrib(gen);
|
||||||
set_exponent(&AccDecn, exp, sign);
|
set_exponent(&AccDecn, exp, sign);
|
||||||
|
remove_leading_zeros(&AccDecn);
|
||||||
int lsu0 = AccDecn.lsu[0];
|
int lsu0 = AccDecn.lsu[0];
|
||||||
|
exp = get_exponent(&AccDecn);
|
||||||
CAPTURE(lsu0);
|
CAPTURE(lsu0);
|
||||||
CAPTURE(exp);
|
CAPTURE(exp);
|
||||||
CAPTURE(sign);
|
CAPTURE(sign);
|
||||||
if (exp == -1 && lsu0 == 0){
|
if (exp <= -1){
|
||||||
//very small
|
//very small
|
||||||
sin_test(40);
|
sin_test(4000);
|
||||||
} else if ((exp == -1 && lsu0 < 10) || (exp == 0 && lsu0 == 0)){
|
} else if (exp == 0 && lsu0 < 50){
|
||||||
//small
|
//small
|
||||||
sin_test(0.4);
|
sin_test(0.4);
|
||||||
} else if ((exp == 0 && lsu0 == 31)){
|
} else if (exp == 2 && lsu0 >= 17 && lsu0 <= 19){
|
||||||
//near pi
|
//near 180
|
||||||
sin_test(0.2);
|
sin_test(3);
|
||||||
} else if ((exp == 0 && lsu0 == 62)){
|
} else if (exp == 2 && lsu0 >= 35 && lsu0 <= 36){
|
||||||
//near 2pi
|
//near 360
|
||||||
sin_test(0.2);
|
sin_test(12);
|
||||||
} else if ((exp == 0 && lsu0 > 62)){
|
} else if (exp == 2 && lsu0 > 50){
|
||||||
//large
|
//large
|
||||||
sin_test(0.1);
|
sin_test(35);
|
||||||
} else {
|
} else {
|
||||||
sin_test(0.02);
|
sin_test(0.02);
|
||||||
}
|
}
|
||||||
@ -302,7 +380,7 @@ TEST_CASE("sin random"){
|
|||||||
TEST_CASE("cos random"){
|
TEST_CASE("cos random"){
|
||||||
std::default_random_engine gen;
|
std::default_random_engine gen;
|
||||||
std::uniform_int_distribution<int> distrib(0,99);
|
std::uniform_int_distribution<int> distrib(0,99);
|
||||||
std::uniform_int_distribution<int> exp_distrib(-1,0); //restrict range for now
|
std::uniform_int_distribution<int> exp_distrib(0, 2); //restrict range for now
|
||||||
std::uniform_int_distribution<int> sign_distrib(0,1);
|
std::uniform_int_distribution<int> sign_distrib(0,1);
|
||||||
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
||||||
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
||||||
@ -311,20 +389,24 @@ TEST_CASE("cos random"){
|
|||||||
int exp = exp_distrib(gen);
|
int exp = exp_distrib(gen);
|
||||||
int sign = sign_distrib(gen);
|
int sign = sign_distrib(gen);
|
||||||
set_exponent(&AccDecn, exp, sign);
|
set_exponent(&AccDecn, exp, sign);
|
||||||
|
remove_leading_zeros(&AccDecn);
|
||||||
int lsu0 = AccDecn.lsu[0];
|
int lsu0 = AccDecn.lsu[0];
|
||||||
|
exp = get_exponent(&AccDecn);
|
||||||
CAPTURE(lsu0);
|
CAPTURE(lsu0);
|
||||||
CAPTURE(exp);
|
CAPTURE(exp);
|
||||||
CAPTURE(sign);
|
CAPTURE(sign);
|
||||||
if (exp == 0 && lsu0 == 15){
|
if (exp == 1 && lsu0 >= 89 && lsu0 <= 90){
|
||||||
//near pi/2
|
//very near 90
|
||||||
cos_test(0.4);
|
cos_test(500);
|
||||||
} else if (exp == 0 && lsu0 == 47){
|
} else if (exp == 1 && lsu0 >= 87 && lsu0 <= 92){
|
||||||
//near 3/2 * pi
|
//near 90
|
||||||
cos_test(0.4);
|
cos_test(2);
|
||||||
} else if (exp == 0 && lsu0 == 78){
|
} else if (exp == 2 && lsu0 >= 26 && lsu0 <= 27){
|
||||||
//near 5/2 * pi
|
//near 270
|
||||||
// cos_test(0.4);
|
cos_test(500);
|
||||||
cos_test(1.1); //actual rtol is much worse than 0.4, random test happens to hit a bad one
|
} else if (exp == 2 && lsu0 >= 44){
|
||||||
|
//large
|
||||||
|
cos_test(20);
|
||||||
} else {
|
} else {
|
||||||
cos_test(0.02);
|
cos_test(0.02);
|
||||||
}
|
}
|
||||||
@ -334,7 +416,7 @@ TEST_CASE("cos random"){
|
|||||||
TEST_CASE("tan random"){
|
TEST_CASE("tan random"){
|
||||||
std::default_random_engine gen;
|
std::default_random_engine gen;
|
||||||
std::uniform_int_distribution<int> distrib(0,99);
|
std::uniform_int_distribution<int> distrib(0,99);
|
||||||
std::uniform_int_distribution<int> exp_distrib(-1,0); //restrict range for now
|
std::uniform_int_distribution<int> exp_distrib(0, 2); //restrict range for now
|
||||||
std::uniform_int_distribution<int> sign_distrib(0,1);
|
std::uniform_int_distribution<int> sign_distrib(0,1);
|
||||||
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
for (int j = 0; j < NUM_RAND_TRIG_TESTS; j++){
|
||||||
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
for (int i = 0; i < DEC80_NUM_LSU; i++){
|
||||||
@ -343,35 +425,39 @@ TEST_CASE("tan random"){
|
|||||||
int exp = exp_distrib(gen);
|
int exp = exp_distrib(gen);
|
||||||
int sign = sign_distrib(gen);
|
int sign = sign_distrib(gen);
|
||||||
set_exponent(&AccDecn, exp, sign);
|
set_exponent(&AccDecn, exp, sign);
|
||||||
|
remove_leading_zeros(&AccDecn);
|
||||||
int lsu0 = AccDecn.lsu[0];
|
int lsu0 = AccDecn.lsu[0];
|
||||||
|
exp = get_exponent(&AccDecn);
|
||||||
CAPTURE(lsu0);
|
CAPTURE(lsu0);
|
||||||
CAPTURE(exp);
|
CAPTURE(exp);
|
||||||
CAPTURE(sign);
|
CAPTURE(sign);
|
||||||
if (exp == -1 && lsu0 == 0){
|
if (exp <= -3){
|
||||||
|
//extremely small
|
||||||
|
tan_test(5000);
|
||||||
|
} if (exp <= -1){
|
||||||
//very small
|
//very small
|
||||||
tan_test(40);
|
tan_test(400);
|
||||||
} else if ((exp == -1 && lsu0 < 10) || (exp == 0 && lsu0 == 0)){
|
} else if (exp == 0 && lsu0 < 50){
|
||||||
//small
|
//small
|
||||||
tan_test(0.5);
|
tan_test(1);
|
||||||
} else if (exp == 0 && lsu0 == 15){
|
} else if (exp == 1 && lsu0 >= 89 && lsu0 <= 90){
|
||||||
//near pi/2
|
//very near 90
|
||||||
tan_test(0.5);
|
tan_test(5);
|
||||||
} else if ((exp == 0 && lsu0 == 31)){
|
} else if (exp == 1 && lsu0 >= 87 && lsu0 <= 92){
|
||||||
//near pi
|
//near 90
|
||||||
tan_test(0.2);
|
tan_test(1);
|
||||||
} else if (exp == 0 && lsu0 == 47){
|
} else if (exp == 2 && lsu0 >= 17 && lsu0 <= 19){
|
||||||
//near 3/2 * pi
|
//near 180
|
||||||
tan_test(0.5);
|
tan_test(3);
|
||||||
} else if ((exp == 0 && lsu0 == 62)){
|
} else if (exp == 2 && lsu0 >= 26 && lsu0 <= 27){
|
||||||
//near 2pi
|
//near 270
|
||||||
tan_test(0.2);
|
tan_test(5);
|
||||||
} else if (exp == 0 && lsu0 == 78){
|
} else if (exp == 2 && lsu0 >= 35 && lsu0 <= 37){
|
||||||
//near 5/2 * pi
|
//near 360
|
||||||
// tan_test(0.5);
|
tan_test(20);
|
||||||
tan_test(0.6); //actual rtol is much worse than 0.4, random test happens to hit a bad one
|
} else if (exp == 2 && lsu0 >= 44){
|
||||||
} else if ((exp == 0 && lsu0 > 62)){
|
|
||||||
//large
|
//large
|
||||||
tan_test(0.1);
|
tan_test(50);
|
||||||
} else {
|
} else {
|
||||||
tan_test(0.02);
|
tan_test(0.02);
|
||||||
}
|
}
|
||||||
|
@ -117,7 +117,7 @@ static char readBusy() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void wait_busy() {
|
static void wait_busy() {
|
||||||
unsigned int i;
|
uint8_t i;
|
||||||
for (i = 0; i < 100; i++){
|
for (i = 0; i < 100; i++){
|
||||||
if (!readBusy()){
|
if (!readBusy()){
|
||||||
return;
|
return;
|
||||||
@ -129,7 +129,6 @@ static void wait_busy() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static void LCD_OutChar(unsigned char c) {
|
static void LCD_OutChar(unsigned char c) {
|
||||||
unsigned char lower = (c & 0x0f);
|
|
||||||
DISABLE_INTERRUPTS();
|
DISABLE_INTERRUPTS();
|
||||||
wait_busy();
|
wait_busy();
|
||||||
//output upper 4 bits:
|
//output upper 4 bits:
|
||||||
|
@ -140,7 +140,7 @@ static void latch_on(void)
|
|||||||
|
|
||||||
__xdata char EntryBuf[MAX_CHARS_PER_LINE + 1];
|
__xdata char EntryBuf[MAX_CHARS_PER_LINE + 1];
|
||||||
__xdata uint8_t ExpBuf[2];
|
__xdata uint8_t ExpBuf[2];
|
||||||
__code const char VER_STR[32+1] = "STC RPN Calculator v1.12";
|
__code const char VER_STR[32+1] = "STC RPN Calculator v1.13";
|
||||||
|
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
|
Loading…
Reference in New Issue
Block a user