The next part of this crazy project is to scale up the 8-bit random numbers to 16-bit. An LFSR basically works by a LSL/EOR combo, and the 16-bit version is no different.
; AVR ASM program to make "random" 16-bit numbers
; using LFSR with a seed, and two spanners!
; A N Peck Sat 21 Aug 2021 11:02:58 AEST
.nolist
.include "tn13Adef.inc" ; Define device ATtiny13A
.list
.equ span1= 23 ; count before spanner1 thrown
.equ spanum1= 7 ; size of spanner1
.def spanner1= r18 ; spanner1 counting register
.equ span2= 37 ; count before spanner2 thrown
.equ spanum2= 3 ; size of spanner2
.def spanner2= r19 ; spanner2 counting register
.equ seed= 2901 ; starting seed - happy birthday
.equ poly1= 13 ; change these
.equ poly2= 24 ; polynomials if need
.dseg
.org SRAM_START
.cseg
.org 000000
rjmp Main ; Reset vector
Main:
; initialise stack
ldi r16, Low(RAMEND)
out SPL, r16
; seed for LFSR loaded
ldi r16, low(seed)
mov r7, r16
ldi r16, high(seed)
mov r6, r16
; polynomial loaded
ldi r24, poly1
ldi r25, poly2
; spanners loaded
ldi r18, span1
ldi r19, span2
Loop:
; generate random number
rcall rand_16
checkspanner1:
subi spanner1, 1 ; decrement spanner1 count
brne nospanner ; branch if not at zero
mov r16, r7 ; throw spanner1 (subtraction)
subi r16, spanum1
breq nozero1 ; result must be nonzero
mov r7, r16
nozero1:
ldi spanner1, span1 ; reset spanner1 count
checkspanner2:
subi spanner2, 1 ; decrement spanner2 count
brne nospanner ; branch if not at zero
mov r16, r6 ; throw spanner2 (addition)
subi r16, -spanum2
breq nozero2 ; result must be nonzero
mov r6, r16
nozero2:
ldi spanner2, span2 ; reset spanner2 count
nospanner:
; save registers to file with modified nop
nop
rjmp loop
rand_16:
lsl r6 ; shift first
rol r7 ; roll
brcc noxor ; check flag
eor r6, r24 ; XOR if needed
eor r7, r25
noxor:
ret
To test the randomness of the generated numbers I used a special version of Gerd's AVR Simulator (thanks Gerd!) which allowed me to capture the register values in a csv file. I then used CONCATENATE and HEX2DEC in a spreadsheet to make a long list of decimal values in the range 0 to 65535 (around 6000 values).
Finally I imported this data as raw input into Audacity and then listened to the noise resulting.
This allowed me to tinker with the code until a "white noise" signal resulted - I especially worked two "spanners" into the code whereby after a certain pre-determined number of numbers the random number was "jumped" up or down by a specific amount. Crude, but effective.
Another option would be to swap nibbles at "random" points (I may still code this).
The next part of this project is to see how easy it is to incorporate assembly code such as this (based on the ATTiny13 AVR instruction set) to the PFS154. Watch this space.
For such a long time I resisted the lure of the logic chip - why should I bother when a microcontroller can do all that and more for little more than spare change?
Well, now that microcontrollers (if you can buy them) cost millions of dollars each and logic chips are still relatively cheap - that side of the equation has changed!
But also I have to admit I've fallen a little in love with these "wire critters". One element of the change of heart has been reading Forrest Mims' excellent notebook series, many of which contain fascinating circuits based on single or combinations of different logic chips.
There are many useful circuits (non-microcontroller based) that can be constructed, such as the following button debouncing circuit made with the CD4001 (see last blog and video):
While playing around with some ideas for applications of "latching" circuits I decided that a long held issue of mine found in most school science laboratories may be addressed with a circuit involving this IC.
The problem has been that there is only one big red panic button in the average school laboratory used in an emergency to shutdown gas, electricity and/or water. I'd like to have one panic button at each "station" which can shut down the entire grid.
To accomplish this I built a simulated version and it seemed to work both for a single station, but also for multiple stations as you would find in an actual laboratory.
The simulation worked fine, so then I constructed the circuit on a breadboard and ran it through its paces.
The lab tech at work reckons that the idea is a winner, but is worried about students pushing the big red button just to annoy everyone. That is an entirely possible (probable) scenario, but can easily be addressed with the usual carrot/stick training that we employ for other "dangerous" or tricky laboratory items.
I've always been more of a coder learning electronics than an electronics peep learning coding, but I'm coming around to the notion of CMOS logic chips doing the work for simple circuits - particularly with the price of micro-controllers at the moment.
For around AU$0.06 each the CD4001 has a lot of handy uses. Using 0033mer as inspiration, I decided to make a classic relaxation oscillator feeding into a flasher suitable for car indicators.
It's an ingenious little circuit (as is usual with 0033mer) and it worked perfectly the first time. It does make me wonder if the indicators on cars use a similar system and, if not, will we return to logic driven circuits in cars as the micro-controller market continues convulsing?
I spent quite a few happy hours fiddling around with traffic light changers after my last blog and video which used the PFS154 and a button to make the changes. The crazy part is that there is not one single traffic light on my 35 minute commute to work - so...huh?
Anyway, first I looked at an interrupt driven change on the ATTiny13 version, then a simple loop driven change for an ESP32.
BTW, there are so many incarnations of the ESP32 that the list covers several pages on my Arduino-IDE.
After choosing the appropriate variant in the IDE I coded the following three versions:
1. Button press activated
2. WiFi activated
3. Bluetooth activated
Button Press:
int red = 13;
int yellow = 12;
int green = 14;
int button = 23;
int buttonValue = 0;
voidsetup(){
pinMode(button,INPUT);
pinMode(red,OUTPUT);
pinMode(yellow,OUTPUT);
pinMode(green,OUTPUT);
pinMode(button,INPUT);
digitalWrite(red,HIGH);
}
voidloop(){
buttonValue = digitalRead(button);
if (buttonValue == HIGH){
changeLights();
delay(4000);
digitalWrite(green, LOW);
digitalWrite(yellow, HIGH);
delay(1000);
digitalWrite(yellow, LOW);
digitalWrite(red, HIGH);
delay(2000);
buttonValue = LOW;
}
}
voidchangeLights(){
// red off, yellow for 1 seconds
digitalWrite(red, LOW);
digitalWrite(yellow, HIGH);
delay(1000);
// turn off yellow, then turn green on
digitalWrite(yellow, LOW);
digitalWrite(green, HIGH);
}
// the following are the RGB leds for the C3
constbyte red = 3;
constbyte green = 4;
constbyte blue = 5;
#define SERVICE_UUID "6E400001-B5A3-F393-E0A9-E50E24DCCA9E"#define CHARACTERISTIC_UUID_RX "6E400002-B5A3-F393-E0A9-E50E24DCCA9E"#define CHARACTERISTIC_UUID_TX "6E400003-B5A3-F393-E0A9-E50E24DCCA9E"voidchangeLights() {
digitalWrite(red, HIGH);
digitalWrite(green, HIGH);
digitalWrite(blue, LOW);
delay(2000);
digitalWrite(red, LOW);
digitalWrite(green, HIGH);
digitalWrite(blue, LOW);
delay(4000);
digitalWrite(red, HIGH);
digitalWrite(green, HIGH);
digitalWrite(blue, LOW);
delay(2000);
digitalWrite(red, HIGH);
digitalWrite(green, LOW);
digitalWrite(blue, LOW);
}
classMyServerCallbacks: public BLEServerCallbacks {
voidonConnect(BLEServer* pServer) {
deviceConnected = true;
};
voidonDisconnect(BLEServer* pServer) {
deviceConnected = false;
}
};
classMyCallbacks: public BLECharacteristicCallbacks {
voidonWrite(BLECharacteristic *pCharacteristic) {
std::string rxValue = pCharacteristic->getValue();
if (rxValue.length() > 0) {
for (int i = 0; i < rxValue.length(); i++) {
}
if (rxValue.find("G") != -1) {
changeLights();
}
}
}
};
voidsetup() {
pinMode(red, OUTPUT);
pinMode(green, OUTPUT);
pinMode(blue, OUTPUT);
digitalWrite(red, HIGH);
digitalWrite(green, LOW);
digitalWrite(blue, LOW);
BLEDevice::init("Traffic Light Changer");
BLEServer *pServer = BLEDevice::createServer(); pServer->setCallbacks(new MyServerCallbacks());
BLEService *pService = pServer->createService(SERVICE_UUID);
pCharacteristic = pService->createCharacteristic(
CHARACTERISTIC_UUID_TX,
BLECharacteristic::PROPERTY_NOTIFY
);
pCharacteristic->addDescriptor(new BLE2902());
BLECharacteristic *pCharacteristic = pService->createCharacteristic(
CHARACTERISTIC_UUID_RX,
BLECharacteristic::PROPERTY_WRITE
);
pCharacteristic->setCallbacks(new MyCallbacks());
pService->start();
pServer->getAdvertising()->start();
}
voidloop() {
if (deviceConnected) {
}
delay(1000);
}
Given the distinct dearth of traffic lights in Tasmania, I'm not sure how directly useful any of these devices will be in shortening my daily commute, but I can say that it opens up some possibilities for monitoring sensors and activating motors/pumps, etc, around the property. So perhaps overall it was a useful exercise!?
I also "enjoyed" the wrestle with the C3-BLE version of the ESP32, which I think may be very interesting to play with in the future.
The world is a wacky place at times - and right now it is difficult to get a relatively common IC like the STM32. Supply these days is often an issue, and price is certainly becoming an issue.
In September 2020 I bought 10 STM32F103R8T6 ICs to make the Pesky Padauk Programmer for the very reasonable cost of AUD $11.30 (for all 10) including delivery. In September 2021 I spied the following at my usual "cheap" supplier:
Yikes! So when a new PCB Padauk Programmer based on Tim's design turned up recently, I thought I'd desolder the STM32 chips from some of my previous numpty versions and try them in the new design.
Result? More or less successful! I think some of those previously used STM32s were fried from my amateur attempts to make this programmer. They have tiny fragile pins, and I have gross fine motor skills instead of the required fine gross motor skills!
At some point I will also want to, for the "intellectual stimulation", desolder a STM32F103R8T6 and/or STM32F103R6T6 from a "blue pill" to see if I can make that work, but that will have to be the subject of another video and blog I think!